function runTest(config, qualifier) { var prefix = testnamePrefix(qualifier, config.keysystem) + ', requestMediaKeySystemAccess: '; function expect_error(keySystem, configurations, expectedError, testname) { var audioCapabilities = configurations.length ? configurations[0].audioCapabilities : undefined, videoCapabilities = configurations.length ? configurations[0].videoCapabilities : undefined, audiocontenttypes = audioCapabilities ? audioCapabilities.map( function(ac) { return "'" + ac.contentType + "'"; } ).join(',') : '', videocontenttypes = videoCapabilities ? videoCapabilities.map( function(ac) { return "'" + ac.contentType + "'"; } ).join(',') : '', modifiedtestname = testname.replace( '%audiocontenttype', audiocontenttypes ).replace( '%videocontenttype', videocontenttypes ); promise_test(function(test) { var p = navigator.requestMediaKeySystemAccess(keySystem, configurations); // expectedError is a string name for the error. We can differentiate // JS Errors from DOMExceptions by checking whether // window[expectedError] exists. If it does, expectedError is the name // of a JS Error subclass and window[expectedError] is the constructor // for that subclass. Otherwise it's a name for a DOMException. if (window[expectedError]) { return promise_rejects_js(test, window[expectedError], p); } else { return promise_rejects_dom(test, expectedError, p); } }, prefix + modifiedtestname + ' should result in ' + expectedError ); } function assert_subset(actual, expected, path) { if (typeof expected == 'string') { assert_equals(actual, expected, path); } else { if (expected.hasOwnProperty('length')) { assert_equals(actual.length, expected.length, path + '.length'); } for (property in expected) { assert_subset(actual[property], expected[property], path + '.' + property); } } } function expect_config(keySystem, configurations, expectedConfiguration, testname) { promise_test(function(test) { return navigator.requestMediaKeySystemAccess(keySystem, configurations).then(function(a) { assert_subset(a.getConfiguration(), expectedConfiguration, testname + ': '); }); }, testname); } // Tests for Key System. expect_error('', [{}], 'TypeError', 'Empty Key System'); expect_error('com.example.unsupported', [{}], 'NotSupportedError', 'Unsupported Key System'); expect_error(config.keysystem + '.', [{}], 'NotSupportedError', 'Key System ending in "."'); expect_error(config.keysystem.toUpperCase(), [{}], 'NotSupportedError', 'Capitalized Key System'); expect_error(config.keysystem + '\u028F', [{}], 'NotSupportedError', 'Non-ASCII Key System'); // Parent of Clear Key not supported. expect_error(config.keysystem.match(/^(.*?)\./)[1], [{}], 'NotSupportedError', 'Root domain of Key System alone'); expect_error(config.keysystem.match(/^(.*?)\./)[0], [{}], 'NotSupportedError', 'Root domain of Key System, with dot'); expect_error(config.keysystem.match(/^(.*?\..*?)\./)[1], [{}], 'NotSupportedError', 'Domain of Key System along'); expect_error(config.keysystem.match(/^(.*?\..*?)\./)[0], [{}], 'NotSupportedError', 'Domain of Key System, with dot'); // Child of Clear Key not supported. expect_error(config.keysystem+'.foo', [{}], 'NotSupportedError', 'Child of Key System'); // Prefixed Clear Key not supported. expect_error('webkit-'+config.keysystem, [{}], 'NotSupportedError', 'Prefixed Key System'); // Incomplete names. expect_error(config.keysystem.substr(0,7)+config.keysystem.substr(8), [{}], 'NotSupportedError', 'Missing characters in middle of Key System name'); expect_error(config.keysystem.substr(0,config.keysystem.length-1), [{}], 'NotSupportedError', 'Missing characters at end of Key System name'); // Spaces in key system name not supported. expect_error(' '+config.keysystem, [{}], 'NotSupportedError', 'Leading space in Key System name'); expect_error(config.keysystem.substr(0,6) + ' ' + config.keysystem.substr(6), [{}], 'NotSupportedError', 'Extra space in Key System name'); expect_error(config.keysystem + ' ', [{}], 'NotSupportedError', 'Trailing space in Key System name'); // Extra dots in key systems names not supported. expect_error('.' + config.keysystem, [{}], 'NotSupportedError', 'Leading dot in Key System name'); expect_error(config.keysystem.substr(0,6) + '.' + config.keysystem.substr(6), [{}], 'NotSupportedError', 'Extra dot in middle of Key System name'); expect_error(config.keysystem + '.', [{}], 'NotSupportedError', 'Trailing dot in Key System name'); // Key system name is case sensitive. if (config.keysystem !== config.keysystem.toUpperCase()) { expect_error(config.keysystem.toUpperCase(), [{}], 'NotSupportedError', 'Key System name is case sensitive'); } if (config.keysystem !== config.keysystem.toLowerCase()) { expect_error(config.keysystem.toLowerCase(), [{}], 'NotSupportedError', 'Key System name is case sensitive'); } // Tests for trivial configurations. expect_error(config.keysystem, [], 'TypeError', 'Empty supportedConfigurations'); expect_error(config.keysystem, [{}], 'NotSupportedError', 'Empty configuration'); // Various combinations of supportedConfigurations. expect_config(config.keysystem, [{ initDataTypes: [config.initDataType], audioCapabilities: [{contentType: config.audioType}], videoCapabilities: [{contentType: config.videoType}], label: 'abcd', }], { initDataTypes: [config.initDataType], audioCapabilities: [{contentType: config.audioType}], videoCapabilities: [{contentType: config.videoType}], label: 'abcd', }, 'Basic supported configuration'); expect_config(config.keysystem, [{ initDataTypes: ['fakeidt', config.initDataType], audioCapabilities: [{contentType: 'audio/fake'}, {contentType: config.audioType}], videoCapabilities: [{contentType: 'video/fake'}, {contentType: config.videoType}], }], { initDataTypes: [config.initDataType], audioCapabilities: [{contentType: config.audioType}], videoCapabilities: [{contentType: config.videoType}], }, 'Partially supported configuration'); expect_config(config.keysystem, [{ audioCapabilities: [{contentType: config.audioType}], }], { audioCapabilities: [{contentType: config.audioType}], }, 'Supported audio codec'); expect_config(config.keysystem, [{ audioCapabilities: [{contentType: config.audioType.replace(/^(.*?);(.*)/, "$1; $2")}], }], { audioCapabilities: [{contentType: config.audioType.replace(/^(.*?);(.*)/, "$1; $2")}], }, 'ContentType formatting must be preserved'); expect_error(config.keysystem, [{ audioCapabilities: [{contentType: 'audio/webm; codecs=fake'}], }], 'NotSupportedError', 'Unsupported audio codec (%audiocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [{contentType: 'video/webm; codecs=fake'}], }], 'NotSupportedError', 'Unsupported video codec (%videocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/webm; codecs=mp4a'}, {contentType: 'audio/webm; codecs=mp4a.40.2'} ], }], 'NotSupportedError', 'Mismatched audio container/codec (%audiocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [{contentType: config.videoType}], }], 'NotSupportedError', 'Video codec specified in audio field (%audiocontenttype)'); expect_error(config.keysystem, [{ videoCapabilities: [{contentType: config.audioType}], }], 'NotSupportedError', 'Audio codec specified in video field (%videocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/webm; codecs=avc1'}, {contentType: 'audio/webm; codecs=avc1.42e01e'} ], }], 'NotSupportedError', 'Mismatched audio container/codec (%audiocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/mp4; codecs=vorbis'} ], }], 'NotSupportedError', 'Mismatched audio container/codec (%audiocontenttype)'); expect_config(config.keysystem, [{ initDataTypes: ['fakeidt'], videoCapabilities: [{contentType: config.videoType}] }, { initDataTypes: [config.initDataType], videoCapabilities: [{contentType: config.videoType}] } ], { initDataTypes: [config.initDataType], videoCapabilities: [{contentType: config.videoType}] }, 'Two configurations, one supported'); expect_config(config.keysystem, [{ initDataTypes: [config.initDataType], videoCapabilities: [{contentType: config.videoType}] }, { videoCapabilities: [{contentType: config.videoType}] } ], { initDataTypes: [config.initDataType], videoCapabilities: [{contentType: config.videoType}] }, 'Two configurations, both supported'); // Audio MIME type does not support video codecs. expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/webm; codecs="vp8,vorbis"'}, {contentType: 'audio/webm; codecs="vorbis, vp8"'}, {contentType: 'audio/webm; codecs="vp8"'} ], }], 'NotSupportedError', 'Audio MIME type does not support video codecs (webm) (%audiocontenttype)'); expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/mp4; codecs="avc1"'}, {contentType: 'audio/mp4; codecs="avc1.4d401e"'}, ], }], 'NotSupportedError', 'Audio MIME type does not support video codecs (mp4) (%audiocontenttype)'); // Video MIME type does not support audio codecs. expect_error(config.keysystem, [{ videoCapabilities: [ {contentType: 'video/webm; codecs="vp8,vorbis"'}, {contentType: 'video/webm; codecs="vorbis, vp8"'}, {contentType: 'video/webm; codecs="vorbis"'} ], }], 'NotSupportedError', 'Video MIME type does not support audio codecs (webm) (%videocontenttype)'); expect_error(config.keysystem, [{ videoCapabilities: [ {contentType: 'video/mp4; codecs="mp4a"'}, {contentType: 'video/mp4; codecs="mp4a.40.2"'} ], }], 'NotSupportedError', 'Video MIME type does not support audio codecs (mp4) (%videocontenttype)'); // WebM does not support AVC1/AAC. expect_error(config.keysystem, [{ audioCapabilities: [ {contentType: 'audio/webm; codecs="aac"'}, {contentType: 'audio/webm; codecs="avc1"'}, {contentType: 'audio/webm; codecs="vp8,aac"'} ], }], 'NotSupportedError', 'WebM audio does not support AVC1/AAC (%audiocontenttype)'); expect_error(config.keysystem, [{ videoCapabilities: [ {contentType: 'video/webm; codecs="aac"'}, {contentType: 'video/webm; codecs="avc1"'}, {contentType: 'video/webm; codecs="vp8,aac"'} ], }], 'NotSupportedError', 'WebM video does not support AVC1/AAC (%videocontenttype)'); // Extra space is allowed in contentType. expect_config(config.keysystem, [{ videoCapabilities: [{contentType: ' ' + config.videoType}], }], { videoCapabilities: [{contentType: ' ' + config.videoType}], }, 'Leading space in contentType'); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType.replace( /^(.*?);(.*)/, "$1 ;$2")}], }], { videoCapabilities: [{contentType: config.videoType.replace( /^(.*?);(.*)/, "$1 ;$2")}], }, 'Space before ; in contentType'); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType + ' '}], }], { videoCapabilities: [{contentType: config.videoType + ' '}], }, 'Trailing space in contentType'); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType.replace( /^(.*?codecs=\")(.*)/, "$1 $2")}], }], { videoCapabilities: [{contentType: config.videoType.replace( /^(.*?codecs=\")(.*)/, "$1 $2")}], }, 'Space at start of codecs parameter'); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType.replace( /^(.*?codecs=\".*)\"/, "$1 \"")}], }], { videoCapabilities: [{contentType: config.videoType.replace( /^(.*?codecs=\".*)\"/, "$1 \"")}], }, 'Space at end of codecs parameter'); // contentType is not case sensitive (except the codec names). expect_config(config.keysystem, [{ videoCapabilities: [{contentType: 'V' + config.videoType.substr(1)}], }], { videoCapabilities: [{contentType: 'V' + config.videoType.substr(1)}], }, 'Video/' ); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType.replace( /^(.*?)c(odecs.*)/, "$1C$2")}], }], { videoCapabilities: [{contentType: config.videoType.replace( /^(.*?)c(odecs.*)/, "$1C$2")}], }, 'Codecs='); var t = config.videoType.match(/(.*?)(;.*)/); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: t[1].toUpperCase() + t[2]}], }], { videoCapabilities: [{contentType: t[1].toUpperCase() + t[2]}], }, 'Upper case MIME type'); t = config.videoType.match(/(.*?)codecs(.*)/); expect_config(config.keysystem, [{ videoCapabilities: [{contentType: t[1] + 'CODECS' + t[2]}], }], { videoCapabilities: [{contentType: t[1] + 'CODECS' + t[2]}], }, 'CODECS='); // Unrecognized attributes are not allowed. expect_error(config.keysystem, [{ videoCapabilities: [{contentType: 'video/webm; foo="bar"'}], }], 'NotSupportedError', 'Unrecognized foo with webm (%videocontenttype)'); expect_error(config.keysystem, [{ videoCapabilities: [{contentType: 'video/mp4; foo="bar"'}], }], 'NotSupportedError', 'Unrecognized foo with mp4 (%videocontenttype)'); expect_error(config.keysystem, [{ videoCapabilities: [{contentType: config.videoType + '; foo="bar"'}], }], 'NotSupportedError', 'Unrecognized foo with codecs (%videocontenttype)'); // Invalid contentTypes. expect_error(config.keysystem, [{ videoCapabilities: [{contentType: 'fake'}], }], 'NotSupportedError', 'contentType: %videocontenttype'); expect_error(config.keysystem, [{ audioCapabilities: [{contentType: 'audio/fake'}], }], 'NotSupportedError', 'contentType: %audiocontenttype'); expect_error(config.keysystem, [{ videoCapabilities: [{contentType: 'video/fake'}], }], 'NotSupportedError', 'contentType: %videocontenttype'); // The actual codec names are case sensitive. t = config.videoType.match( /(.*?codecs=\")(.*?\")(.*)/ ); if (t[2] !== t[2].toUpperCase()) { expect_error(config.keysystem, [{ videoCapabilities: [{contentType: t[1] + t[2].toUpperCase() + t[3] }], }], 'NotSupportedError', 'contentType: %videocontenttype'); } if (t[2] !== t[2].toLowerCase()) { expect_error(config.keysystem, [{ videoCapabilities: [{contentType: t[1] + t[2].toLowerCase() + t[3] }], }], 'NotSupportedError', 'contentType: %videocontenttype'); } // Extra comma is not allowed in codecs. expect_error(config.keysystem, [{ videoCapabilities: [{contentType: t[1] + ',' + t[2] + t[3] }], }], 'NotSupportedError', 'contentType: %videocontenttype'); }