diff options
Diffstat (limited to 'testing/web-platform/tests/resources/idlharness.js')
-rw-r--r-- | testing/web-platform/tests/resources/idlharness.js | 1852 |
1 files changed, 0 insertions, 1852 deletions
diff --git a/testing/web-platform/tests/resources/idlharness.js b/testing/web-platform/tests/resources/idlharness.js deleted file mode 100644 index 314223001..000000000 --- a/testing/web-platform/tests/resources/idlharness.js +++ /dev/null @@ -1,1852 +0,0 @@ -/* -Distributed under both the W3C Test Suite License [1] and the W3C -3-clause BSD License [2]. To contribute to a W3C Test Suite, see the -policies and contribution forms [3]. - -[1] http://www.w3.org/Consortium/Legal/2008/04-testsuite-license -[2] http://www.w3.org/Consortium/Legal/2008/03-bsd-license -[3] http://www.w3.org/2004/10/27-testcases -*/ - -/* For user documentation see docs/idlharness.md */ - -/** - * Notes for people who want to edit this file (not just use it as a library): - * - * Most of the interesting stuff happens in the derived classes of IdlObject, - * especially IdlInterface. The entry point for all IdlObjects is .test(), - * which is called by IdlArray.test(). An IdlObject is conceptually just - * "thing we want to run tests on", and an IdlArray is an array of IdlObjects - * with some additional data thrown in. - * - * The object model is based on what WebIDLParser.js produces, which is in turn - * based on its pegjs grammar. If you want to figure out what properties an - * object will have from WebIDLParser.js, the best way is to look at the - * grammar: - * - * https://github.com/darobin/webidl.js/blob/master/lib/grammar.peg - * - * So for instance: - * - * // interface definition - * interface - * = extAttrs:extendedAttributeList? S? "interface" S name:identifier w herit:ifInheritance? w "{" w mem:ifMember* w "}" w ";" w - * { return { type: "interface", name: name, inheritance: herit, members: mem, extAttrs: extAttrs }; } - * - * This means that an "interface" object will have a .type property equal to - * the string "interface", a .name property equal to the identifier that the - * parser found, an .inheritance property equal to either null or the result of - * the "ifInheritance" production found elsewhere in the grammar, and so on. - * After each grammatical production is a JavaScript function in curly braces - * that gets called with suitable arguments and returns some JavaScript value. - * - * (Note that the version of WebIDLParser.js we use might sometimes be - * out-of-date or forked.) - * - * The members and methods of the classes defined by this file are all at least - * briefly documented, hopefully. - */ -(function(){ -"use strict"; -/// Helpers /// -function constValue (cnt) { - if (cnt.type === "null") return null; - if (cnt.type === "NaN") return NaN; - if (cnt.type === "Infinity") return cnt.negative ? -Infinity : Infinity; - return cnt.value; -} - -function minOverloadLength(overloads) { - if (!overloads.length) { - return 0; - } - - return overloads.map(function(attr) { - return attr.arguments ? attr.arguments.filter(function(arg) { - return !arg.optional && !arg.variadic; - }).length : 0; - }) - .reduce(function(m, n) { return Math.min(m, n); }); -} - -function throwOrReject(a_test, operation, fn, obj, args, message, cb) { - if (operation.idlType.generic !== "Promise") { - assert_throws(new TypeError(), function() { - fn.apply(obj, args); - }, message); - cb(); - } else { - try { - promise_rejects(a_test, new TypeError(), fn.apply(obj, args)).then(cb, cb); - } catch (e){ - a_test.step(function() { - assert_unreached("Throws \"" + e + "\" instead of rejecting promise"); - cb(); - }); - } - } -} - -function awaitNCallbacks(n, cb, ctx) { - var counter = 0; - return function() { - counter++; - if (counter >= n) { - cb(); - } - }; -} - -var fround = (function(){ - if (Math.fround) return Math.fround; - - var arr = new Float32Array(1); - return function fround(n) { - arr[0] = n; - return arr[0]; - }; -})(); - -/// IdlArray /// -// Entry point -self.IdlArray = function() -//@{ -{ - /** - * A map from strings to the corresponding named IdlObject, such as - * IdlInterface or IdlException. These are the things that test() will run - * tests on. - */ - this.members = {}; - - /** - * A map from strings to arrays of strings. The keys are interface or - * exception names, and are expected to also exist as keys in this.members - * (otherwise they'll be ignored). This is populated by add_objects() -- - * see documentation at the start of the file. The actual tests will be - * run by calling this.members[name].test_object(obj) for each obj in - * this.objects[name]. obj is a string that will be eval'd to produce a - * JavaScript value, which is supposed to be an object implementing the - * given IdlObject (interface, exception, etc.). - */ - this.objects = {}; - - /** - * When adding multiple collections of IDLs one at a time, an earlier one - * might contain a partial interface or implements statement that depends - * on a later one. Save these up and handle them right before we run - * tests. - * - * .partials is simply an array of objects from WebIDLParser.js' - * "partialinterface" production. .implements maps strings to arrays of - * strings, such that - * - * A implements B; - * A implements C; - * D implements E; - * - * results in { A: ["B", "C"], D: ["E"] }. - */ - this.partials = []; - this["implements"] = {}; -}; - -//@} -IdlArray.prototype.add_idls = function(raw_idls) -//@{ -{ - /** Entry point. See documentation at beginning of file. */ - this.internal_add_idls(WebIDL2.parse(raw_idls)); -}; - -//@} -IdlArray.prototype.add_untested_idls = function(raw_idls) -//@{ -{ - /** Entry point. See documentation at beginning of file. */ - var parsed_idls = WebIDL2.parse(raw_idls); - for (var i = 0; i < parsed_idls.length; i++) - { - parsed_idls[i].untested = true; - if ("members" in parsed_idls[i]) - { - for (var j = 0; j < parsed_idls[i].members.length; j++) - { - parsed_idls[i].members[j].untested = true; - } - } - } - this.internal_add_idls(parsed_idls); -}; - -//@} -IdlArray.prototype.internal_add_idls = function(parsed_idls) -//@{ -{ - /** - * Internal helper called by add_idls() and add_untested_idls(). - * parsed_idls is an array of objects that come from WebIDLParser.js's - * "definitions" production. The add_untested_idls() entry point - * additionally sets an .untested property on each object (and its - * .members) so that they'll be skipped by test() -- they'll only be - * used for base interfaces of tested interfaces, return types, etc. - */ - parsed_idls.forEach(function(parsed_idl) - { - if (parsed_idl.type == "interface" && parsed_idl.partial) - { - this.partials.push(parsed_idl); - return; - } - - if (parsed_idl.type == "implements") - { - if (!(parsed_idl.target in this["implements"])) - { - this["implements"][parsed_idl.target] = []; - } - this["implements"][parsed_idl.target].push(parsed_idl["implements"]); - return; - } - - parsed_idl.array = this; - if (parsed_idl.name in this.members) - { - throw "Duplicate identifier " + parsed_idl.name; - } - switch(parsed_idl.type) - { - case "interface": - this.members[parsed_idl.name] = - new IdlInterface(parsed_idl, /* is_callback = */ false); - break; - - case "dictionary": - // Nothing to test, but we need the dictionary info around for type - // checks - this.members[parsed_idl.name] = new IdlDictionary(parsed_idl); - break; - - case "typedef": - this.members[parsed_idl.name] = new IdlTypedef(parsed_idl); - break; - - case "callback": - // TODO - console.log("callback not yet supported"); - break; - - case "enum": - this.members[parsed_idl.name] = new IdlEnum(parsed_idl); - break; - - case "callback interface": - this.members[parsed_idl.name] = - new IdlInterface(parsed_idl, /* is_callback = */ true); - break; - - default: - throw parsed_idl.name + ": " + parsed_idl.type + " not yet supported"; - } - }.bind(this)); -}; - -//@} -IdlArray.prototype.add_objects = function(dict) -//@{ -{ - /** Entry point. See documentation at beginning of file. */ - for (var k in dict) - { - if (k in this.objects) - { - this.objects[k] = this.objects[k].concat(dict[k]); - } - else - { - this.objects[k] = dict[k]; - } - } -}; - -//@} -IdlArray.prototype.prevent_multiple_testing = function(name) -//@{ -{ - /** Entry point. See documentation at beginning of file. */ - this.members[name].prevent_multiple_testing = true; -}; - -//@} -IdlArray.prototype.recursively_get_implements = function(interface_name) -//@{ -{ - /** - * Helper function for test(). Returns an array of things that implement - * interface_name, so if the IDL contains - * - * A implements B; - * B implements C; - * B implements D; - * - * then recursively_get_implements("A") should return ["B", "C", "D"]. - */ - var ret = this["implements"][interface_name]; - if (ret === undefined) - { - return []; - } - for (var i = 0; i < this["implements"][interface_name].length; i++) - { - ret = ret.concat(this.recursively_get_implements(ret[i])); - if (ret.indexOf(ret[i]) != ret.lastIndexOf(ret[i])) - { - throw "Circular implements statements involving " + ret[i]; - } - } - return ret; -}; - -function exposed_in(globals) { - if ('document' in self) { - return globals.indexOf("Window") >= 0; - } - if ('DedicatedWorkerGlobalScope' in self && - self instanceof DedicatedWorkerGlobalScope) { - return globals.indexOf("Worker") >= 0 || - globals.indexOf("DedicatedWorker") >= 0; - } - if ('SharedWorkerGlobalScope' in self && - self instanceof SharedWorkerGlobalScope) { - return globals.indexOf("Worker") >= 0 || - globals.indexOf("SharedWorker") >= 0; - } - if ('ServiceWorkerGlobalScope' in self && - self instanceof ServiceWorkerGlobalScope) { - return globals.indexOf("Worker") >= 0 || - globals.indexOf("ServiceWorker") >= 0; - } - throw "Unexpected global object"; -} - -//@} -IdlArray.prototype.test = function() -//@{ -{ - /** Entry point. See documentation at beginning of file. */ - - // First merge in all the partial interfaces and implements statements we - // encountered. - this.partials.forEach(function(parsed_idl) - { - if (!(parsed_idl.name in this.members) - || !(this.members[parsed_idl.name] instanceof IdlInterface)) - { - throw "Partial interface " + parsed_idl.name + " with no original interface"; - } - if (parsed_idl.extAttrs) - { - parsed_idl.extAttrs.forEach(function(extAttr) - { - this.members[parsed_idl.name].extAttrs.push(extAttr); - }.bind(this)); - } - parsed_idl.members.forEach(function(member) - { - this.members[parsed_idl.name].members.push(new IdlInterfaceMember(member)); - }.bind(this)); - }.bind(this)); - this.partials = []; - - for (var lhs in this["implements"]) - { - this.recursively_get_implements(lhs).forEach(function(rhs) - { - var errStr = lhs + " implements " + rhs + ", but "; - if (!(lhs in this.members)) throw errStr + lhs + " is undefined."; - if (!(this.members[lhs] instanceof IdlInterface)) throw errStr + lhs + " is not an interface."; - if (!(rhs in this.members)) throw errStr + rhs + " is undefined."; - if (!(this.members[rhs] instanceof IdlInterface)) throw errStr + rhs + " is not an interface."; - this.members[rhs].members.forEach(function(member) - { - this.members[lhs].members.push(new IdlInterfaceMember(member)); - }.bind(this)); - }.bind(this)); - } - this["implements"] = {}; - - Object.getOwnPropertyNames(this.members).forEach(function(memberName) { - var member = this.members[memberName]; - if (!(member instanceof IdlInterface)) { - return; - } - - var exposed = member.extAttrs.filter(function(a) { return a.name == "Exposed" }); - if (exposed.length > 1) { - throw "Unexpected Exposed extended attributes on " + memberName + ": " + exposed; - } - - var globals = exposed.length === 1 - ? exposed[0].rhs.value - : ["Window"]; - member.exposed = exposed_in(globals); - }.bind(this)); - - // Now run test() on every member, and test_object() for every object. - for (var name in this.members) - { - this.members[name].test(); - if (name in this.objects) - { - this.objects[name].forEach(function(str) - { - this.members[name].test_object(str); - }.bind(this)); - } - } -}; - -//@} -IdlArray.prototype.assert_type_is = function(value, type) -//@{ -{ - /** - * Helper function that tests that value is an instance of type according - * to the rules of WebIDL. value is any JavaScript value, and type is an - * object produced by WebIDLParser.js' "type" production. That production - * is fairly elaborate due to the complexity of WebIDL's types, so it's - * best to look at the grammar to figure out what properties it might have. - */ - if (type.idlType == "any") - { - // No assertions to make - return; - } - - if (type.nullable && value === null) - { - // This is fine - return; - } - - if (type.array) - { - // TODO: not supported yet - return; - } - - if (type.sequence) - { - assert_true(Array.isArray(value), "is not array"); - if (!value.length) - { - // Nothing we can do. - return; - } - this.assert_type_is(value[0], type.idlType.idlType); - return; - } - - type = type.idlType; - - switch(type) - { - case "void": - assert_equals(value, undefined); - return; - - case "boolean": - assert_equals(typeof value, "boolean"); - return; - - case "byte": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(-128 <= value && value <= 127, "byte " + value + " not in range [-128, 127]"); - return; - - case "octet": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(0 <= value && value <= 255, "octet " + value + " not in range [0, 255]"); - return; - - case "short": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(-32768 <= value && value <= 32767, "short " + value + " not in range [-32768, 32767]"); - return; - - case "unsigned short": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(0 <= value && value <= 65535, "unsigned short " + value + " not in range [0, 65535]"); - return; - - case "long": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(-2147483648 <= value && value <= 2147483647, "long " + value + " not in range [-2147483648, 2147483647]"); - return; - - case "unsigned long": - assert_equals(typeof value, "number"); - assert_equals(value, Math.floor(value), "not an integer"); - assert_true(0 <= value && value <= 4294967295, "unsigned long " + value + " not in range [0, 4294967295]"); - return; - - case "long long": - assert_equals(typeof value, "number"); - return; - - case "unsigned long long": - case "DOMTimeStamp": - assert_equals(typeof value, "number"); - assert_true(0 <= value, "unsigned long long is negative"); - return; - - case "float": - assert_equals(typeof value, "number"); - assert_equals(value, fround(value), "float rounded to 32-bit float should be itself"); - assert_not_equals(value, Infinity); - assert_not_equals(value, -Infinity); - assert_not_equals(value, NaN); - return; - - case "DOMHighResTimeStamp": - case "double": - assert_equals(typeof value, "number"); - assert_not_equals(value, Infinity); - assert_not_equals(value, -Infinity); - assert_not_equals(value, NaN); - return; - - case "unrestricted float": - assert_equals(typeof value, "number"); - assert_equals(value, fround(value), "unrestricted float rounded to 32-bit float should be itself"); - return; - - case "unrestricted double": - assert_equals(typeof value, "number"); - return; - - case "DOMString": - assert_equals(typeof value, "string"); - return; - - case "ByteString": - assert_equals(typeof value, "string"); - assert_regexp_match(value, /^[\x00-\x7F]*$/); - return; - - case "USVString": - assert_equals(typeof value, "string"); - assert_regexp_match(value, /^([\x00-\ud7ff\ue000-\uffff]|[\ud800-\udbff][\udc00-\udfff])*$/); - return; - - case "object": - assert_true(typeof value == "object" || typeof value == "function", "wrong type: not object or function"); - return; - } - - if (!(type in this.members)) - { - throw "Unrecognized type " + type; - } - - if (this.members[type] instanceof IdlInterface) - { - // We don't want to run the full - // IdlInterface.prototype.test_instance_of, because that could result - // in an infinite loop. TODO: This means we don't have tests for - // NoInterfaceObject interfaces, and we also can't test objects that - // come from another self. - assert_true(typeof value == "object" || typeof value == "function", "wrong type: not object or function"); - if (value instanceof Object - && !this.members[type].has_extended_attribute("NoInterfaceObject") - && type in self) - { - assert_true(value instanceof self[type], "not instanceof " + type); - } - } - else if (this.members[type] instanceof IdlEnum) - { - assert_equals(typeof value, "string"); - } - else if (this.members[type] instanceof IdlDictionary) - { - // TODO: Test when we actually have something to test this on - } - else if (this.members[type] instanceof IdlTypedef) - { - // TODO: Test when we actually have something to test this on - } - else - { - throw "Type " + type + " isn't an interface or dictionary"; - } -}; -//@} - -/// IdlObject /// -function IdlObject() {} -IdlObject.prototype.test = function() -//@{ -{ - /** - * By default, this does nothing, so no actual tests are run for IdlObjects - * that don't define any (e.g., IdlDictionary at the time of this writing). - */ -}; - -//@} -IdlObject.prototype.has_extended_attribute = function(name) -//@{ -{ - /** - * This is only meaningful for things that support extended attributes, - * such as interfaces, exceptions, and members. - */ - return this.extAttrs.some(function(o) - { - return o.name == name; - }); -}; - -//@} - -/// IdlDictionary /// -// Used for IdlArray.prototype.assert_type_is -function IdlDictionary(obj) -//@{ -{ - /** - * obj is an object produced by the WebIDLParser.js "dictionary" - * production. - */ - - /** Self-explanatory. */ - this.name = obj.name; - - /** An array of objects produced by the "dictionaryMember" production. */ - this.members = obj.members; - - /** - * The name (as a string) of the dictionary type we inherit from, or null - * if there is none. - */ - this.base = obj.inheritance; -} - -//@} -IdlDictionary.prototype = Object.create(IdlObject.prototype); - -/// IdlInterface /// -function IdlInterface(obj, is_callback) { - /** - * obj is an object produced by the WebIDLParser.js "interface" production. - */ - - /** Self-explanatory. */ - this.name = obj.name; - - /** A back-reference to our IdlArray. */ - this.array = obj.array; - - /** - * An indicator of whether we should run tests on the interface object and - * interface prototype object. Tests on members are controlled by .untested - * on each member, not this. - */ - this.untested = obj.untested; - - /** An array of objects produced by the "ExtAttr" production. */ - this.extAttrs = obj.extAttrs; - - /** An array of IdlInterfaceMembers. */ - this.members = obj.members.map(function(m){return new IdlInterfaceMember(m); }); - if (this.has_extended_attribute("Unforgeable")) { - this.members - .filter(function(m) { return !m["static"] && (m.type == "attribute" || m.type == "operation"); }) - .forEach(function(m) { return m.isUnforgeable = true; }); - } - - /** - * The name (as a string) of the type we inherit from, or null if there is - * none. - */ - this.base = obj.inheritance; - - this._is_callback = is_callback; -} -IdlInterface.prototype = Object.create(IdlObject.prototype); -IdlInterface.prototype.is_callback = function() -//@{ -{ - return this._is_callback; -}; -//@} - -IdlInterface.prototype.has_constants = function() -//@{ -{ - return this.members.some(function(member) { - return member.type === "const"; - }); -}; -//@} - -IdlInterface.prototype.is_global = function() -//@{ -{ - return this.extAttrs.some(function(attribute) { - return attribute.name === "Global" || - attribute.name === "PrimaryGlobal"; - }); -}; -//@} - -IdlInterface.prototype.test = function() -//@{ -{ - if (this.has_extended_attribute("NoInterfaceObject")) - { - // No tests to do without an instance. TODO: We should still be able - // to run tests on the prototype object, if we obtain one through some - // other means. - return; - } - - if (!this.exposed) { - test(function() { - assert_false(this.name in self); - }.bind(this), this.name + " interface: existence and properties of interface object"); - return; - } - - if (!this.untested) - { - // First test things to do with the exception/interface object and - // exception/interface prototype object. - this.test_self(); - } - // Then test things to do with its members (constants, fields, attributes, - // operations, . . .). These are run even if .untested is true, because - // members might themselves be marked as .untested. This might happen to - // interfaces if the interface itself is untested but a partial interface - // that extends it is tested -- then the interface itself and its initial - // members will be marked as untested, but the members added by the partial - // interface are still tested. - this.test_members(); -}; -//@} - -IdlInterface.prototype.test_self = function() -//@{ -{ - test(function() - { - // This function tests WebIDL as of 2015-01-13. - - // "For every interface that is exposed in a given ECMAScript global - // environment and: - // * is a callback interface that has constants declared on it, or - // * is a non-callback interface that is not declared with the - // [NoInterfaceObject] extended attribute, - // a corresponding property MUST exist on the ECMAScript global object. - // The name of the property is the identifier of the interface, and its - // value is an object called the interface object. - // The property has the attributes { [[Writable]]: true, - // [[Enumerable]]: false, [[Configurable]]: true }." - if (this.is_callback() && !this.has_constants()) { - return; - } - - // TODO: Should we test here that the property is actually writable - // etc., or trust getOwnPropertyDescriptor? - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - var desc = Object.getOwnPropertyDescriptor(self, this.name); - assert_false("get" in desc, "self's property " + format_value(this.name) + " has getter"); - assert_false("set" in desc, "self's property " + format_value(this.name) + " has setter"); - assert_true(desc.writable, "self's property " + format_value(this.name) + " is not writable"); - assert_false(desc.enumerable, "self's property " + format_value(this.name) + " is enumerable"); - assert_true(desc.configurable, "self's property " + format_value(this.name) + " is not configurable"); - - if (this.is_callback()) { - // "The internal [[Prototype]] property of an interface object for - // a callback interface MUST be the Object.prototype object." - assert_equals(Object.getPrototypeOf(self[this.name]), Object.prototype, - "prototype of self's property " + format_value(this.name) + " is not Object.prototype"); - - return; - } - - // "The interface object for a given non-callback interface is a - // function object." - // "If an object is defined to be a function object, then it has - // characteristics as follows:" - - // Its [[Prototype]] internal property is otherwise specified (see - // below). - - // "* Its [[Get]] internal property is set as described in ECMA-262 - // section 9.1.8." - // Not much to test for this. - - // "* Its [[Construct]] internal property is set as described in - // ECMA-262 section 19.2.2.3." - // Tested below if no constructor is defined. TODO: test constructors - // if defined. - - // "* Its @@hasInstance property is set as described in ECMA-262 - // section 19.2.3.8, unless otherwise specified." - // TODO - - // ES6 (rev 30) 19.1.3.6: - // "Else, if O has a [[Call]] internal method, then let builtinTag be - // "Function"." - assert_class_string(self[this.name], "Function", "class string of " + this.name); - - // "The [[Prototype]] internal property of an interface object for a - // non-callback interface is determined as follows:" - var prototype = Object.getPrototypeOf(self[this.name]); - if (this.base) { - // "* If the interface inherits from some other interface, the - // value of [[Prototype]] is the interface object for that other - // interface." - var has_interface_object = - !this.array - .members[this.base] - .has_extended_attribute("NoInterfaceObject"); - if (has_interface_object) { - assert_own_property(self, this.base, - 'should inherit from ' + this.base + - ', but self has no such property'); - assert_equals(prototype, self[this.base], - 'prototype of ' + this.name + ' is not ' + - this.base); - } - } else { - // "If the interface doesn't inherit from any other interface, the - // value of [[Prototype]] is %FunctionPrototype% ([ECMA-262], - // section 6.1.7.4)." - assert_equals(prototype, Function.prototype, - "prototype of self's property " + format_value(this.name) + " is not Function.prototype"); - } - - if (!this.has_extended_attribute("Constructor")) { - // "The internal [[Call]] method of the interface object behaves as - // follows . . . - // - // "If I was not declared with a [Constructor] extended attribute, - // then throw a TypeError." - assert_throws(new TypeError(), function() { - self[this.name](); - }.bind(this), "interface object didn't throw TypeError when called as a function"); - assert_throws(new TypeError(), function() { - new self[this.name](); - }.bind(this), "interface object didn't throw TypeError when called as a constructor"); - } - }.bind(this), this.name + " interface: existence and properties of interface object"); - - if (!this.is_callback()) { - test(function() { - // This function tests WebIDL as of 2014-10-25. - // https://heycam.github.io/webidl/#es-interface-call - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - // "Interface objects for non-callback interfaces MUST have a - // property named “length” with attributes { [[Writable]]: false, - // [[Enumerable]]: false, [[Configurable]]: true } whose value is - // a Number." - assert_own_property(self[this.name], "length"); - var desc = Object.getOwnPropertyDescriptor(self[this.name], "length"); - assert_false("get" in desc, this.name + ".length has getter"); - assert_false("set" in desc, this.name + ".length has setter"); - assert_false(desc.writable, this.name + ".length is writable"); - assert_false(desc.enumerable, this.name + ".length is enumerable"); - assert_true(desc.configurable, this.name + ".length is not configurable"); - - var constructors = this.extAttrs - .filter(function(attr) { return attr.name == "Constructor"; }); - var expected_length = minOverloadLength(constructors); - assert_equals(self[this.name].length, expected_length, "wrong value for " + this.name + ".length"); - }.bind(this), this.name + " interface object length"); - } - - if (!this.is_callback() || this.has_constants()) { - test(function() { - // This function tests WebIDL as of 2015-11-17. - // https://heycam.github.io/webidl/#interface-object - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - // "All interface objects must have a property named “name” with - // attributes { [[Writable]]: false, [[Enumerable]]: false, - // [[Configurable]]: true } whose value is the identifier of the - // corresponding interface." - - assert_own_property(self[this.name], "name"); - var desc = Object.getOwnPropertyDescriptor(self[this.name], "name"); - assert_false("get" in desc, this.name + ".name has getter"); - assert_false("set" in desc, this.name + ".name has setter"); - assert_false(desc.writable, this.name + ".name is writable"); - assert_false(desc.enumerable, this.name + ".name is enumerable"); - assert_true(desc.configurable, this.name + ".name is not configurable"); - assert_equals(self[this.name].name, this.name, "wrong value for " + this.name + ".name"); - }.bind(this), this.name + " interface object name"); - } - - // TODO: Test named constructors if I find any interfaces that have them. - - test(function() - { - // This function tests WebIDL as of 2015-01-21. - // https://heycam.github.io/webidl/#interface-object - - if (this.is_callback() && !this.has_constants()) { - return; - } - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - if (this.is_callback()) { - assert_false("prototype" in self[this.name], - this.name + ' should not have a "prototype" property'); - return; - } - - // "An interface object for a non-callback interface must have a - // property named “prototype” with attributes { [[Writable]]: false, - // [[Enumerable]]: false, [[Configurable]]: false } whose value is an - // object called the interface prototype object. This object has - // properties that correspond to the regular attributes and regular - // operations defined on the interface, and is described in more detail - // in section 4.5.4 below." - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - var desc = Object.getOwnPropertyDescriptor(self[this.name], "prototype"); - assert_false("get" in desc, this.name + ".prototype has getter"); - assert_false("set" in desc, this.name + ".prototype has setter"); - assert_false(desc.writable, this.name + ".prototype is writable"); - assert_false(desc.enumerable, this.name + ".prototype is enumerable"); - assert_false(desc.configurable, this.name + ".prototype is configurable"); - - // Next, test that the [[Prototype]] of the interface prototype object - // is correct. (This is made somewhat difficult by the existence of - // [NoInterfaceObject].) - // TODO: Aryeh thinks there's at least other place in this file where - // we try to figure out if an interface prototype object is - // correct. Consolidate that code. - - // "The interface prototype object for a given interface A must have an - // internal [[Prototype]] property whose value is returned from the - // following steps: - // "If A is declared with the [Global] or [PrimaryGlobal] extended - // attribute, and A supports named properties, then return the named - // properties object for A, as defined in section 4.5.5 below. - // "Otherwise, if A is declared to inherit from another interface, then - // return the interface prototype object for the inherited interface. - // "Otherwise, if A is declared with the [ArrayClass] extended - // attribute, then return %ArrayPrototype% ([ECMA-262], section - // 6.1.7.4). - // "Otherwise, return %ObjectPrototype% ([ECMA-262], section 6.1.7.4). - // ([ECMA-262], section 15.2.4). - if (this.name === "Window") { - assert_class_string(Object.getPrototypeOf(self[this.name].prototype), - 'WindowProperties', - 'Class name for prototype of Window' + - '.prototype is not "WindowProperties"'); - } else { - var inherit_interface, inherit_interface_has_interface_object; - if (this.base) { - inherit_interface = this.base; - inherit_interface_has_interface_object = - !this.array - .members[inherit_interface] - .has_extended_attribute("NoInterfaceObject"); - } else if (this.has_extended_attribute('ArrayClass')) { - inherit_interface = 'Array'; - inherit_interface_has_interface_object = true; - } else { - inherit_interface = 'Object'; - inherit_interface_has_interface_object = true; - } - if (inherit_interface_has_interface_object) { - assert_own_property(self, inherit_interface, - 'should inherit from ' + inherit_interface + ', but self has no such property'); - assert_own_property(self[inherit_interface], 'prototype', - 'should inherit from ' + inherit_interface + ', but that object has no "prototype" property'); - assert_equals(Object.getPrototypeOf(self[this.name].prototype), - self[inherit_interface].prototype, - 'prototype of ' + this.name + '.prototype is not ' + inherit_interface + '.prototype'); - } else { - // We can't test that we get the correct object, because this is the - // only way to get our hands on it. We only test that its class - // string, at least, is correct. - assert_class_string(Object.getPrototypeOf(self[this.name].prototype), - inherit_interface + 'Prototype', - 'Class name for prototype of ' + this.name + - '.prototype is not "' + inherit_interface + 'Prototype"'); - } - } - - // "The class string of an interface prototype object is the - // concatenation of the interface’s identifier and the string - // “Prototype”." - assert_class_string(self[this.name].prototype, this.name + "Prototype", - "class string of " + this.name + ".prototype"); - // String() should end up calling {}.toString if nothing defines a - // stringifier. - if (!this.has_stringifier()) { - assert_equals(String(self[this.name].prototype), "[object " + this.name + "Prototype]", - "String(" + this.name + ".prototype)"); - } - }.bind(this), this.name + " interface: existence and properties of interface prototype object"); - - test(function() - { - if (this.is_callback() && !this.has_constants()) { - return; - } - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - if (this.is_callback()) { - assert_false("prototype" in self[this.name], - this.name + ' should not have a "prototype" property'); - return; - } - - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - // "If the [NoInterfaceObject] extended attribute was not specified on - // the interface, then the interface prototype object must also have a - // property named “constructor” with attributes { [[Writable]]: true, - // [[Enumerable]]: false, [[Configurable]]: true } whose value is a - // reference to the interface object for the interface." - assert_own_property(self[this.name].prototype, "constructor", - this.name + '.prototype does not have own property "constructor"'); - var desc = Object.getOwnPropertyDescriptor(self[this.name].prototype, "constructor"); - assert_false("get" in desc, this.name + ".prototype.constructor has getter"); - assert_false("set" in desc, this.name + ".prototype.constructor has setter"); - assert_true(desc.writable, this.name + ".prototype.constructor is not writable"); - assert_false(desc.enumerable, this.name + ".prototype.constructor is enumerable"); - assert_true(desc.configurable, this.name + ".prototype.constructor in not configurable"); - assert_equals(self[this.name].prototype.constructor, self[this.name], - this.name + '.prototype.constructor is not the same object as ' + this.name); - }.bind(this), this.name + ' interface: existence and properties of interface prototype object\'s "constructor" property'); -}; - -//@} -IdlInterface.prototype.test_member_const = function(member) -//@{ -{ - if (!this.has_constants()) { - throw "Internal error: test_member_const called without any constants"; - } - - test(function() - { - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - // "For each constant defined on an interface A, there must be - // a corresponding property on the interface object, if it - // exists." - assert_own_property(self[this.name], member.name); - // "The value of the property is that which is obtained by - // converting the constant’s IDL value to an ECMAScript - // value." - assert_equals(self[this.name][member.name], constValue(member.value), - "property has wrong value"); - // "The property has attributes { [[Writable]]: false, - // [[Enumerable]]: true, [[Configurable]]: false }." - var desc = Object.getOwnPropertyDescriptor(self[this.name], member.name); - assert_false("get" in desc, "property has getter"); - assert_false("set" in desc, "property has setter"); - assert_false(desc.writable, "property is writable"); - assert_true(desc.enumerable, "property is not enumerable"); - assert_false(desc.configurable, "property is configurable"); - }.bind(this), this.name + " interface: constant " + member.name + " on interface object"); - - // "In addition, a property with the same characteristics must - // exist on the interface prototype object." - test(function() - { - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - if (this.is_callback()) { - assert_false("prototype" in self[this.name], - this.name + ' should not have a "prototype" property'); - return; - } - - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - assert_own_property(self[this.name].prototype, member.name); - assert_equals(self[this.name].prototype[member.name], constValue(member.value), - "property has wrong value"); - var desc = Object.getOwnPropertyDescriptor(self[this.name], member.name); - assert_false("get" in desc, "property has getter"); - assert_false("set" in desc, "property has setter"); - assert_false(desc.writable, "property is writable"); - assert_true(desc.enumerable, "property is not enumerable"); - assert_false(desc.configurable, "property is configurable"); - }.bind(this), this.name + " interface: constant " + member.name + " on interface prototype object"); -}; - - -//@} -IdlInterface.prototype.test_member_attribute = function(member) -//@{ -{ - test(function() - { - if (this.is_callback() && !this.has_constants()) { - return; - } - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - if (member["static"]) { - assert_own_property(self[this.name], member.name, - "The interface object must have a property " + - format_value(member.name)); - } else if (this.is_global()) { - assert_own_property(self, member.name, - "The global object must have a property " + - format_value(member.name)); - assert_false(member.name in self[this.name].prototype, - "The prototype object must not have a property " + - format_value(member.name)); - - var getter = Object.getOwnPropertyDescriptor(self, member.name).get; - assert_equals(typeof(getter), "function", - format_value(member.name) + " must have a getter"); - - // Try/catch around the get here, since it can legitimately throw. - // If it does, we obviously can't check for equality with direct - // invocation of the getter. - var gotValue; - var propVal; - try { - propVal = self[member.name]; - gotValue = true; - } catch (e) { - gotValue = false; - } - if (gotValue) { - assert_equals(propVal, getter.call(undefined), - "Gets on a global should not require an explicit this"); - } - - this.do_interface_attribute_asserts(self, member); - } else { - assert_true(member.name in self[this.name].prototype, - "The prototype object must have a property " + - format_value(member.name)); - - if (!member.has_extended_attribute("LenientThis")) { - assert_throws(new TypeError(), function() { - self[this.name].prototype[member.name]; - }.bind(this), "getting property on prototype object must throw TypeError"); - } else { - assert_equals(self[this.name].prototype[member.name], undefined, - "getting property on prototype object must return undefined"); - } - this.do_interface_attribute_asserts(self[this.name].prototype, member); - } - }.bind(this), this.name + " interface: attribute " + member.name); -}; - -//@} -IdlInterface.prototype.test_member_operation = function(member) -//@{ -{ - var a_test = async_test(this.name + " interface: operation " + member.name + - "(" + member.arguments.map( - function(m) {return m.idlType.idlType; } ) - +")"); - a_test.step(function() - { - // This function tests WebIDL as of 2015-12-29. - // https://heycam.github.io/webidl/#es-operations - - if (this.is_callback() && !this.has_constants()) { - a_test.done(); - return; - } - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - if (this.is_callback()) { - assert_false("prototype" in self[this.name], - this.name + ' should not have a "prototype" property'); - a_test.done(); - return; - } - - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - // "For each unique identifier of an exposed operation defined on the - // interface, there must exist a corresponding property, unless the - // effective overload set for that identifier and operation and with an - // argument count of 0 has no entries." - - // TODO: Consider [Exposed]. - - // "The location of the property is determined as follows:" - var memberHolderObject; - // "* If the operation is static, then the property exists on the - // interface object." - if (member["static"]) { - assert_own_property(self[this.name], member.name, - "interface object missing static operation"); - memberHolderObject = self[this.name]; - // "* Otherwise, [...] if the interface was declared with the [Global] - // or [PrimaryGlobal] extended attribute, then the property exists - // on every object that implements the interface." - } else if (this.is_global()) { - assert_own_property(self, member.name, - "global object missing non-static operation"); - memberHolderObject = self; - // "* Otherwise, the property exists solely on the interface’s - // interface prototype object." - } else { - assert_own_property(self[this.name].prototype, member.name, - "interface prototype object missing non-static operation"); - memberHolderObject = self[this.name].prototype; - } - this.do_member_operation_asserts(memberHolderObject, member, a_test); - }.bind(this)); -}; - -//@} -IdlInterface.prototype.do_member_operation_asserts = function(memberHolderObject, member, a_test) -//@{ -{ - var done = a_test.done.bind(a_test); - var operationUnforgeable = member.isUnforgeable; - var desc = Object.getOwnPropertyDescriptor(memberHolderObject, member.name); - // "The property has attributes { [[Writable]]: B, - // [[Enumerable]]: true, [[Configurable]]: B }, where B is false if the - // operation is unforgeable on the interface, and true otherwise". - assert_false("get" in desc, "property has getter"); - assert_false("set" in desc, "property has setter"); - assert_equals(desc.writable, !operationUnforgeable, - "property should be writable if and only if not unforgeable"); - assert_true(desc.enumerable, "property is not enumerable"); - assert_equals(desc.configurable, !operationUnforgeable, - "property should be configurable if and only if not unforgeable"); - // "The value of the property is a Function object whose - // behavior is as follows . . ." - assert_equals(typeof memberHolderObject[member.name], "function", - "property must be a function"); - // "The value of the Function object’s “length” property is - // a Number determined as follows: - // ". . . - // "Return the length of the shortest argument list of the - // entries in S." - assert_equals(memberHolderObject[member.name].length, - minOverloadLength(this.members.filter(function(m) { - return m.type == "operation" && m.name == member.name; - })), - "property has wrong .length"); - - // Make some suitable arguments - var args = member.arguments.map(function(arg) { - return create_suitable_object(arg.idlType); - }); - - // "Let O be a value determined as follows: - // ". . . - // "Otherwise, throw a TypeError." - // This should be hit if the operation is not static, there is - // no [ImplicitThis] attribute, and the this value is null. - // - // TODO: We currently ignore the [ImplicitThis] case. Except we manually - // check for globals, since otherwise we'll invoke window.close(). And we - // have to skip this test for anything that on the proto chain of "self", - // since that does in fact have implicit-this behavior. - if (!member["static"]) { - var cb; - if (!this.is_global() && - memberHolderObject[member.name] != self[member.name]) - { - cb = awaitNCallbacks(2, done); - throwOrReject(a_test, member, memberHolderObject[member.name], null, args, - "calling operation with this = null didn't throw TypeError", cb); - } else { - cb = awaitNCallbacks(1, done); - } - - // ". . . If O is not null and is also not a platform object - // that implements interface I, throw a TypeError." - // - // TODO: Test a platform object that implements some other - // interface. (Have to be sure to get inheritance right.) - throwOrReject(a_test, member, memberHolderObject[member.name], {}, args, - "calling operation with this = {} didn't throw TypeError", cb); - } else { - done(); - } -} - -//@} -IdlInterface.prototype.test_member_stringifier = function(member) -//@{ -{ - test(function() - { - if (this.is_callback() && !this.has_constants()) { - return; - } - - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - - if (this.is_callback()) { - assert_false("prototype" in self[this.name], - this.name + ' should not have a "prototype" property'); - return; - } - - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - // ". . . the property exists on the interface prototype object." - var interfacePrototypeObject = self[this.name].prototype; - assert_own_property(self[this.name].prototype, "toString", - "interface prototype object missing non-static operation"); - - var stringifierUnforgeable = member.isUnforgeable; - var desc = Object.getOwnPropertyDescriptor(interfacePrototypeObject, "toString"); - // "The property has attributes { [[Writable]]: B, - // [[Enumerable]]: true, [[Configurable]]: B }, where B is false if the - // stringifier is unforgeable on the interface, and true otherwise." - assert_false("get" in desc, "property has getter"); - assert_false("set" in desc, "property has setter"); - assert_equals(desc.writable, !stringifierUnforgeable, - "property should be writable if and only if not unforgeable"); - assert_true(desc.enumerable, "property is not enumerable"); - assert_equals(desc.configurable, !stringifierUnforgeable, - "property should be configurable if and only if not unforgeable"); - // "The value of the property is a Function object, which behaves as - // follows . . ." - assert_equals(typeof interfacePrototypeObject.toString, "function", - "property must be a function"); - // "The value of the Function object’s “length” property is the Number - // value 0." - assert_equals(interfacePrototypeObject.toString.length, 0, - "property has wrong .length"); - - // "Let O be the result of calling ToObject on the this value." - assert_throws(new TypeError(), function() { - self[this.name].prototype.toString.apply(null, []); - }, "calling stringifier with this = null didn't throw TypeError"); - - // "If O is not an object that implements the interface on which the - // stringifier was declared, then throw a TypeError." - // - // TODO: Test a platform object that implements some other - // interface. (Have to be sure to get inheritance right.) - assert_throws(new TypeError(), function() { - self[this.name].prototype.toString.apply({}, []); - }, "calling stringifier with this = {} didn't throw TypeError"); - }.bind(this), this.name + " interface: stringifier"); -}; - -//@} -IdlInterface.prototype.test_members = function() -//@{ -{ - for (var i = 0; i < this.members.length; i++) - { - var member = this.members[i]; - if (member.untested) { - continue; - } - - switch (member.type) { - case "const": - this.test_member_const(member); - break; - - case "attribute": - // For unforgeable attributes, we do the checks in - // test_interface_of instead. - if (!member.isUnforgeable) - { - this.test_member_attribute(member); - } - if (member.stringifier) { - this.test_member_stringifier(member); - } - break; - - case "operation": - // TODO: Need to correctly handle multiple operations with the same - // identifier. - // For unforgeable operations, we do the checks in - // test_interface_of instead. - if (member.name) { - if (!member.isUnforgeable) - { - this.test_member_operation(member); - } - } else if (member.stringifier) { - this.test_member_stringifier(member); - } - break; - - default: - // TODO: check more member types. - break; - } - } -}; - -//@} -IdlInterface.prototype.test_object = function(desc) -//@{ -{ - var obj, exception = null; - try - { - obj = eval(desc); - } - catch(e) - { - exception = e; - } - - var expected_typeof = - this.members.some(function(member) { return member.legacycaller; }) - ? "function" - : "object"; - - this.test_primary_interface_of(desc, obj, exception, expected_typeof); - var current_interface = this; - while (current_interface) - { - if (!(current_interface.name in this.array.members)) - { - throw "Interface " + current_interface.name + " not found (inherited by " + this.name + ")"; - } - if (current_interface.prevent_multiple_testing && current_interface.already_tested) - { - return; - } - current_interface.test_interface_of(desc, obj, exception, expected_typeof); - current_interface = this.array.members[current_interface.base]; - } -}; - -//@} -IdlInterface.prototype.test_primary_interface_of = function(desc, obj, exception, expected_typeof) -//@{ -{ - // We can't easily test that its prototype is correct if there's no - // interface object, or the object is from a different global environment - // (not instanceof Object). TODO: test in this case that its prototype at - // least looks correct, even if we can't test that it's actually correct. - if (!this.has_extended_attribute("NoInterfaceObject") - && (typeof obj != expected_typeof || obj instanceof Object)) - { - test(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - assert_own_property(self, this.name, - "self does not have own property " + format_value(this.name)); - assert_own_property(self[this.name], "prototype", - 'interface "' + this.name + '" does not have own property "prototype"'); - - // "The value of the internal [[Prototype]] property of the - // platform object is the interface prototype object of the primary - // interface from the platform object’s associated global - // environment." - assert_equals(Object.getPrototypeOf(obj), - self[this.name].prototype, - desc + "'s prototype is not " + this.name + ".prototype"); - }.bind(this), this.name + " must be primary interface of " + desc); - } - - // "The class string of a platform object that implements one or more - // interfaces must be the identifier of the primary interface of the - // platform object." - test(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - assert_class_string(obj, this.name, "class string of " + desc); - if (!this.has_stringifier()) - { - assert_equals(String(obj), "[object " + this.name + "]", "String(" + desc + ")"); - } - }.bind(this), "Stringification of " + desc); -}; - -//@} -IdlInterface.prototype.test_interface_of = function(desc, obj, exception, expected_typeof) -//@{ -{ - // TODO: Indexed and named properties, more checks on interface members - this.already_tested = true; - - for (var i = 0; i < this.members.length; i++) - { - var member = this.members[i]; - if (member.type == "attribute" && member.isUnforgeable) - { - test(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - this.do_interface_attribute_asserts(obj, member); - }.bind(this), this.name + " interface: " + desc + ' must have own property "' + member.name + '"'); - } - else if (member.type == "operation" && - member.name && - member.isUnforgeable) - { - var a_test = async_test(this.name + " interface: " + desc + ' must have own property "' + member.name + '"'); - a_test.step(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - assert_own_property(obj, member.name, - "Doesn't have the unforgeable operation property"); - this.do_member_operation_asserts(obj, member, a_test); - }.bind(this)); - } - else if ((member.type == "const" - || member.type == "attribute" - || member.type == "operation") - && member.name) - { - test(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - if (!member["static"]) { - if (!this.is_global()) { - assert_inherits(obj, member.name); - } else { - assert_own_property(obj, member.name); - } - - if (member.type == "const") - { - assert_equals(obj[member.name], constValue(member.value)); - } - if (member.type == "attribute") - { - // Attributes are accessor properties, so they might - // legitimately throw an exception rather than returning - // anything. - var property, thrown = false; - try - { - property = obj[member.name]; - } - catch (e) - { - thrown = true; - } - if (!thrown) - { - this.array.assert_type_is(property, member.idlType); - } - } - if (member.type == "operation") - { - assert_equals(typeof obj[member.name], "function"); - } - } - }.bind(this), this.name + " interface: " + desc + ' must inherit property "' + member.name + '" with the proper type (' + i + ')'); - } - // TODO: This is wrong if there are multiple operations with the same - // identifier. - // TODO: Test passing arguments of the wrong type. - if (member.type == "operation" && member.name && member.arguments.length) - { - var a_test = async_test( this.name + " interface: calling " + member.name + - "(" + member.arguments.map(function(m) { return m.idlType.idlType; }) + - ") on " + desc + " with too few arguments must throw TypeError"); - a_test.step(function() - { - assert_equals(exception, null, "Unexpected exception when evaluating object"); - assert_equals(typeof obj, expected_typeof, "wrong typeof object"); - if (!member["static"]) { - if (!this.is_global() && !member.isUnforgeable) { - assert_inherits(obj, member.name); - } else { - assert_own_property(obj, member.name); - } - } - else - { - assert_false(member.name in obj); - } - - var minLength = minOverloadLength(this.members.filter(function(m) { - return m.type == "operation" && m.name == member.name; - })); - var args = []; - var cb = awaitNCallbacks(minLength, a_test.done.bind(a_test)); - for (var i = 0; i < minLength; i++) { - throwOrReject(a_test, member, obj[member.name], obj, args, "Called with " + i + " arguments", cb); - - args.push(create_suitable_object(member.arguments[i].idlType)); - } - if (minLength === 0) { - cb(); - } - }.bind(this)); - } - } -}; - -//@} -IdlInterface.prototype.has_stringifier = function() -//@{ -{ - if (this.members.some(function(member) { return member.stringifier; })) { - return true; - } - if (this.base && - this.array.members[this.base].has_stringifier()) { - return true; - } - return false; -}; - -//@} -IdlInterface.prototype.do_interface_attribute_asserts = function(obj, member) -//@{ -{ - // This function tests WebIDL as of 2015-01-27. - // TODO: Consider [Exposed]. - - // This is called by test_member_attribute() with the prototype as obj if - // it is not a global, and the global otherwise, and by test_interface_of() - // with the object as obj. - - // "For each exposed attribute of the interface, whether it was declared on - // the interface itself or one of its consequential interfaces, there MUST - // exist a corresponding property. The characteristics of this property are - // as follows:" - - // "The name of the property is the identifier of the attribute." - assert_own_property(obj, member.name); - - // "The property has attributes { [[Get]]: G, [[Set]]: S, [[Enumerable]]: - // true, [[Configurable]]: configurable }, where: - // "configurable is false if the attribute was declared with the - // [Unforgeable] extended attribute and true otherwise; - // "G is the attribute getter, defined below; and - // "S is the attribute setter, also defined below." - var desc = Object.getOwnPropertyDescriptor(obj, member.name); - assert_false("value" in desc, 'property descriptor has value but is supposed to be accessor'); - assert_false("writable" in desc, 'property descriptor has "writable" field but is supposed to be accessor'); - assert_true(desc.enumerable, "property is not enumerable"); - if (member.isUnforgeable) - { - assert_false(desc.configurable, "[Unforgeable] property must not be configurable"); - } - else - { - assert_true(desc.configurable, "property must be configurable"); - } - - - // "The attribute getter is a Function object whose behavior when invoked - // is as follows:" - assert_equals(typeof desc.get, "function", "getter must be Function"); - - // "If the attribute is a regular attribute, then:" - if (!member["static"]) { - // "If O is not a platform object that implements I, then: - // "If the attribute was specified with the [LenientThis] extended - // attribute, then return undefined. - // "Otherwise, throw a TypeError." - if (!member.has_extended_attribute("LenientThis")) { - assert_throws(new TypeError(), function() { - desc.get.call({}); - }.bind(this), "calling getter on wrong object type must throw TypeError"); - } else { - assert_equals(desc.get.call({}), undefined, - "calling getter on wrong object type must return undefined"); - } - } - - // "The value of the Function object’s “length” property is the Number - // value 0." - assert_equals(desc.get.length, 0, "getter length must be 0"); - - - // TODO: Test calling setter on the interface prototype (should throw - // TypeError in most cases). - if (member.readonly - && !member.has_extended_attribute("PutForwards") - && !member.has_extended_attribute("Replaceable")) - { - // "The attribute setter is undefined if the attribute is declared - // readonly and has neither a [PutForwards] nor a [Replaceable] - // extended attribute declared on it." - assert_equals(desc.set, undefined, "setter must be undefined for readonly attributes"); - } - else - { - // "Otherwise, it is a Function object whose behavior when - // invoked is as follows:" - assert_equals(typeof desc.set, "function", "setter must be function for PutForwards, Replaceable, or non-readonly attributes"); - - // "If the attribute is a regular attribute, then:" - if (!member["static"]) { - // "If /validThis/ is false and the attribute was not specified - // with the [LenientThis] extended attribute, then throw a - // TypeError." - // "If the attribute is declared with a [Replaceable] extended - // attribute, then: ..." - // "If validThis is false, then return." - if (!member.has_extended_attribute("LenientThis")) { - assert_throws(new TypeError(), function() { - desc.set.call({}); - }.bind(this), "calling setter on wrong object type must throw TypeError"); - } else { - assert_equals(desc.set.call({}), undefined, - "calling setter on wrong object type must return undefined"); - } - } - - // "The value of the Function object’s “length” property is the Number - // value 1." - assert_equals(desc.set.length, 1, "setter length must be 1"); - } -} -//@} - -/// IdlInterfaceMember /// -function IdlInterfaceMember(obj) -//@{ -{ - /** - * obj is an object produced by the WebIDLParser.js "ifMember" production. - * We just forward all properties to this object without modification, - * except for special extAttrs handling. - */ - for (var k in obj) - { - this[k] = obj[k]; - } - if (!("extAttrs" in this)) - { - this.extAttrs = []; - } - - this.isUnforgeable = this.has_extended_attribute("Unforgeable"); -} - -//@} -IdlInterfaceMember.prototype = Object.create(IdlObject.prototype); - -/// Internal helper functions /// -function create_suitable_object(type) -//@{ -{ - /** - * type is an object produced by the WebIDLParser.js "type" production. We - * return a JavaScript value that matches the type, if we can figure out - * how. - */ - if (type.nullable) - { - return null; - } - switch (type.idlType) - { - case "any": - case "boolean": - return true; - - case "byte": case "octet": case "short": case "unsigned short": - case "long": case "unsigned long": case "long long": - case "unsigned long long": case "float": case "double": - case "unrestricted float": case "unrestricted double": - return 7; - - case "DOMString": - case "ByteString": - case "USVString": - return "foo"; - - case "object": - return {a: "b"}; - - case "Node": - return document.createTextNode("abc"); - } - return null; -} -//@} - -/// IdlEnum /// -// Used for IdlArray.prototype.assert_type_is -function IdlEnum(obj) -//@{ -{ - /** - * obj is an object produced by the WebIDLParser.js "dictionary" - * production. - */ - - /** Self-explanatory. */ - this.name = obj.name; - - /** An array of values produced by the "enum" production. */ - this.values = obj.values; - -} -//@} - -IdlEnum.prototype = Object.create(IdlObject.prototype); - -/// IdlTypedef /// -// Used for IdlArray.prototype.assert_type_is -function IdlTypedef(obj) -//@{ -{ - /** - * obj is an object produced by the WebIDLParser.js "typedef" - * production. - */ - - /** Self-explanatory. */ - this.name = obj.name; - - /** An array of values produced by the "typedef" production. */ - this.values = obj.values; - -} -//@} - -IdlTypedef.prototype = Object.create(IdlObject.prototype); - -}()); |