mirror of
https://github.com/servo/servo.git
synced 2025-06-06 00:25:37 +00:00
Also update the `html/dom/reflection-metadata.html` test to handle the case where `nonce` does not reflect back to the attribute after an IDL change. Part of https://github.com/servo/servo/issues/4577 Fixes https://github.com/web-platform-tests/wpt/issues/43286 Signed-off-by: Tim van der Lippe <tvanderlippe@gmail.com>
1071 lines
50 KiB
JavaScript
Vendored
1071 lines
50 KiB
JavaScript
Vendored
ReflectionTests = {};
|
||
|
||
ReflectionTests.start = new Date().getTime();
|
||
|
||
/**
|
||
* Resolve the given URL to an absolute URL, relative to the current document's
|
||
* address. There's no API that I know of that exposes this directly, so we
|
||
* actually just create an <a> element, set its href, and stitch together the
|
||
* various properties. Seems to work. We don't try to reimplement the
|
||
* algorithm here, because we're not concerned with its correctness -- we're
|
||
* only testing HTML reflection, not Web Addresses.
|
||
*
|
||
* Return the input if the URL couldn't be resolved, per the spec for
|
||
* reflected URL attributes.
|
||
*
|
||
* It seems like IE9 doesn't implement URL decomposition attributes correctly
|
||
* for <a>, which causes all these tests to fail. Ideally I'd do this in some
|
||
* other way, but the failure does stem from an incorrect implementation of
|
||
* HTML, so I'll leave it alone for now.
|
||
*
|
||
* TODO: This relies on reflection to test reflection, so it could mask bugs.
|
||
* Either get a JS implementation of the "resolve a URL" algorithm, or just
|
||
* specify expected values manually here. It shouldn't be too hard to write
|
||
* special cases for all the values we test.
|
||
*/
|
||
ReflectionTests.resolveUrl = function(url) {
|
||
url = String(url);
|
||
var el = document.createElement("a");
|
||
el.href = url;
|
||
var ret = el.protocol + "//" + el.host + el.pathname + el.search + el.hash;
|
||
if (ret == "//") {
|
||
return url;
|
||
} else {
|
||
return ret;
|
||
}
|
||
};
|
||
|
||
/**
|
||
* The "rules for parsing non-negative integers" from the HTML spec. They're
|
||
* mostly used for reflection, so here seems like as good a place to test them
|
||
* as any. Returns false on error.
|
||
*/
|
||
ReflectionTests.parseNonneg = function(input) {
|
||
var value = this.parseInt(input);
|
||
if (value === false || value < 0) {
|
||
return false;
|
||
}
|
||
return value;
|
||
};
|
||
|
||
/**
|
||
* The "rules for parsing integers" from the HTML spec. Returns false on
|
||
* error.
|
||
*/
|
||
ReflectionTests.parseInt = function(input) {
|
||
var position = 0;
|
||
var sign = 1;
|
||
// Skip whitespace
|
||
while (input.length > position && /^[ \t\n\f\r]$/.test(input[position])) {
|
||
position++;
|
||
}
|
||
if (position >= input.length) {
|
||
return false;
|
||
}
|
||
if (input[position] == "-") {
|
||
sign = -1;
|
||
position++;
|
||
} else if (input[position] == "+") {
|
||
position++;
|
||
}
|
||
if (position >= input.length) {
|
||
return false;
|
||
}
|
||
if (!/^[0-9]$/.test(input[position])) {
|
||
return false;
|
||
}
|
||
var value = 0;
|
||
while (input.length > position && /^[0-9]$/.test(input[position])) {
|
||
value *= 10;
|
||
// Don't use parseInt even for single-digit strings . . .
|
||
value += input.charCodeAt(position) - "0".charCodeAt(0);
|
||
position++;
|
||
}
|
||
if (value === 0) {
|
||
return 0;
|
||
}
|
||
return sign * value;
|
||
};
|
||
|
||
/**
|
||
* The "rules for parsing floating-point number values" from the HTML spec.
|
||
* Returns false on error.
|
||
*/
|
||
ReflectionTests.parseFloat = function(input) {
|
||
var position = 0;
|
||
var value = 1;
|
||
var divisor = 1;
|
||
var exponent = 1;
|
||
// Skip whitespace
|
||
while (input.length > position && /^[ \t\n\f\r]$/.test(input[position])) {
|
||
position++;
|
||
}
|
||
if (position >= input.length) {
|
||
return false;
|
||
}
|
||
if (input[position] == "-") {
|
||
value = -1;
|
||
divisor = -1;
|
||
position++;
|
||
} else if (input[position] == "+") {
|
||
position++;
|
||
}
|
||
if (position >= input.length) {
|
||
return false;
|
||
}
|
||
if (input[position] == "." && position+1 < input.length && /^[0-9]$/.test(input[position+1])) {
|
||
value = 0;
|
||
// Use "else" branches rather than "jump to label fraction"
|
||
} else if (!/^[0-9]$/.test(input[position])) {
|
||
return false;
|
||
} else {
|
||
var val = 0;
|
||
while (input.length > position && /^[0-9]$/.test(input[position])) {
|
||
val *= 10;
|
||
// Don't use parseInt even for single-digit strings . . .
|
||
val += input.charCodeAt(position) - "0".charCodeAt(0);
|
||
position++;
|
||
}
|
||
value *= val;
|
||
}
|
||
// Use nested "if" tests rather than "jump to label conversion" or "skip"
|
||
// Fraction:
|
||
if (input.length > position && input[position] == ".") {
|
||
position++;
|
||
while (input.length > position && /^[0-9]$/.test(input[position])) {
|
||
divisor *= 10;
|
||
// Don't use parseInt even for single-digit strings . . .
|
||
value += (input.charCodeAt(position) - "0".charCodeAt(0)) / divisor;
|
||
position++;
|
||
}
|
||
}
|
||
if (input.length > position && (input[position] == "e" || input[position] == "E")) {
|
||
position++;
|
||
if (input.length > position) {
|
||
if (input[position] == "-") {
|
||
exponent = -1;
|
||
position++;
|
||
} else if (input[position] == "+") {
|
||
position++;
|
||
}
|
||
if (input.length > position && /^[0-9]$/.test(input[position])) {
|
||
var exp = 0;
|
||
do {
|
||
exp *= 10;
|
||
// Don't use parseInt even for single-digit strings . . .
|
||
exp += input.charCodeAt(position) - "0".charCodeAt(0);
|
||
position++;
|
||
} while (input.length > position && /^[0-9]$/.test(input[position]));
|
||
exponent *= exp;
|
||
value *= Math.pow(10, exponent);
|
||
}
|
||
}
|
||
}
|
||
// Conversion:
|
||
if (!Number.isFinite(value)) {
|
||
return false;
|
||
}
|
||
if (value === 0) {
|
||
return 0;
|
||
}
|
||
return value;
|
||
}
|
||
|
||
// Used in initializing typeMap
|
||
var binaryString = "\x00\x01\x02\x03\x04\x05\x06\x07 "
|
||
+ "\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f "
|
||
+ "\x10\x11\x12\x13\x14\x15\x16\x17 "
|
||
+ "\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f ";
|
||
var maxInt = 2147483647;
|
||
var minInt = -2147483648;
|
||
var maxUnsigned = 4294967295;
|
||
|
||
/**
|
||
* Array containing the tests and other information for each type of reflected
|
||
* attribute. Meaning of keys:
|
||
*
|
||
* "jsType": What typeof idlObj[idlName] is supposed to be.
|
||
* "defaultVal": The default value to be returned if the attribute is not
|
||
* present and no default is specifically set for this attribute.
|
||
* "domTests": What values to test with setAttribute().
|
||
* "domExpected": What values to expect with IDL get after setAttribute().
|
||
* Defaults to the same as domTests.
|
||
* "idlTests": What values to test with IDL set. Defaults to domTests.
|
||
* "idlDomExpected": What to expect from getAttribute() after IDL set.
|
||
* Defaults to idlTests.
|
||
* "idlIdlExpected": What to expect from IDL get after IDL set. Defaults to
|
||
* idlDomExpected.
|
||
*
|
||
* Note that all tests/expected values are only baselines, and can be expanded
|
||
* with additional tests hardcoded into the function for particular types if
|
||
* necessary. For example, a special codepath is used for enums, and for
|
||
* IDL setters which throw an exception. null means "defaultVal" is the
|
||
* expected value. Expected DOM values are cast to strings by adding "".
|
||
*
|
||
* TODO: Test strings that aren't valid UTF-16. Desired behavior is not clear
|
||
* here at the time of writing, see
|
||
* http://www.w3.org/Bugs/Public/show_bug.cgi?id=12100
|
||
*
|
||
* TODO: Test deleting an IDL attribute, and maybe doing other fun stuff to it.
|
||
*
|
||
* TODO: Test IDL sets of integer types to out-of-range or other weird values.
|
||
* WebIDL says to wrap, but I'm not sure offhand if that's what we want.
|
||
*
|
||
* TODO: tokenlist, settable tokenlist, limited
|
||
*/
|
||
|
||
|
||
ReflectionTests.typeMap = {
|
||
/**
|
||
* "If a reflecting IDL attribute is a DOMString but doesn't fall into any
|
||
* of the above categories, then the getting and setting must be done in a
|
||
* transparent, case-preserving manner."
|
||
*
|
||
* The data object passed to reflects() can contain an optional key
|
||
* treatNullAsEmptyString, whose value is ignored. If it does contain the
|
||
* key, null will be cast to "" instead of "null", per WebIDL
|
||
* [TreatNullAs=EmptyString].
|
||
*/
|
||
"string": {
|
||
"jsType": "string",
|
||
"defaultVal": "",
|
||
"domTests": ["", " " + binaryString + " foo ", undefined, 7, 1.5, "5%", "+100", ".5", true,
|
||
false, {"test": 6}, NaN, +Infinity, -Infinity, "\0", null,
|
||
{"toString":function(){return "test-toString";}},
|
||
{"valueOf":function(){return "test-valueOf";}, toString:null}
|
||
]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a USVString attribute whose content
|
||
* attribute is defined to contain a URL, then on getting, if the content
|
||
* attribute is absent, the IDL attribute must return the empty string.
|
||
* Otherwise, the IDL attribute must parse the value of the content
|
||
* attribute relative to the element's node document and if that is
|
||
* successful, return the resulting URL string. If parsing fails, then the
|
||
* value of the content attribute must be returned instead, converted to a
|
||
* USVString. On setting, the content attribute must be set to the specified
|
||
* new value."
|
||
*
|
||
* Also HTMLHyperLinkElementUtils href, used by a.href and area.href
|
||
*/
|
||
"url": {
|
||
"jsType": "string",
|
||
"defaultVal": "",
|
||
"domTests": ["", " foo ", "http://site.example/",
|
||
"//site.example/path???@#l", binaryString, undefined, 7, 1.5, "5%", "+100", ".5", true,
|
||
false, {"test": 6}, NaN, +Infinity, -Infinity, "\0", null,
|
||
{"toString":function(){return "test-toString";}},
|
||
{"valueOf":function(){return "test-valueOf";}, toString:null}],
|
||
"domExpected": ReflectionTests.resolveUrl,
|
||
"idlIdlExpected": ReflectionTests.resolveUrl
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a DOMString whose content attribute is
|
||
* an enumerated attribute, and the IDL attribute is limited to only known
|
||
* values, then, on getting, the IDL attribute must return the conforming
|
||
* value associated with the state the attribute is in (in its canonical
|
||
* case), or the empty string if the attribute is in a state that has no
|
||
* associated keyword value; and on setting, if the new value is an ASCII
|
||
* case-insensitive match for one of the keywords given for that attribute,
|
||
* then the content attribute must be set to the conforming value
|
||
* associated with the state that the attribute would be in if set to the
|
||
* given new value, otherwise, if the new value is the empty string, then
|
||
* the content attribute must be removed, otherwise, the content attribute
|
||
* must be set to the given new value."
|
||
*
|
||
* "Some attributes are defined as taking one of a finite set of keywords.
|
||
* Such attributes are called enumerated attributes. The keywords are each
|
||
* defined to map to a particular state (several keywords might map to the
|
||
* same state, in which case some of the keywords are synonyms of each
|
||
* other; additionally, some of the keywords can be said to be
|
||
* non-conforming, and are only in the specification for historical
|
||
* reasons). In addition, two default states can be given. The first is the
|
||
* invalid value default, the second is the missing value default.
|
||
*
|
||
* . . .
|
||
*
|
||
* When the attribute is specified, if its value is an ASCII
|
||
* case-insensitive match for one of the given keywords then that keyword's
|
||
* state is the state that the attribute represents. If the attribute value
|
||
* matches none of the given keywords, but the attribute has an invalid
|
||
* value default, then the attribute represents that state. Otherwise, if
|
||
* the attribute value matches none of the keywords but there is a missing
|
||
* value default state defined, then that is the state represented by the
|
||
* attribute. Otherwise, there is no default, and invalid values must be
|
||
* ignored.
|
||
*
|
||
* When the attribute is not specified, if there is a missing value default
|
||
* state defined, then that is the state represented by the (missing)
|
||
* attribute. Otherwise, the absence of the attribute means that there is
|
||
* no state represented."
|
||
*
|
||
* This is only used for enums that are limited to known values, not other
|
||
* enums (those are treated as generic strings by the spec). The data
|
||
* object passed to reflects() can contain these keys:
|
||
*
|
||
* "defaultVal": missing value default (defaults to "")
|
||
* "invalidVal": invalid value default (defaults to defaultVal)
|
||
* "keywords": array of keywords as given by the spec (required)
|
||
* "nonCanon": dictionary mapping non-canonical values to their
|
||
* canonical equivalents (defaults to {})
|
||
* "isNullable": Indicates if attribute is nullable (defaults to false)
|
||
*
|
||
* Tests are mostly hardcoded into reflects(), since they depend on the
|
||
* keywords. All expected values are computed in reflects() using a helper
|
||
* function.
|
||
*/
|
||
"enum": {
|
||
"jsType": "string",
|
||
"defaultVal": "",
|
||
"domTests": ["", " " + binaryString + " foo ", undefined, 7, 1.5, "5%", "+100", ".5", true,
|
||
false, {"test": 6}, NaN, +Infinity, -Infinity, "\0", null,
|
||
{"toString":function(){return "test-toString";}},
|
||
{"valueOf":function(){return "test-valueOf";}, toString:null}]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a boolean attribute, then on getting
|
||
* the IDL attribute must return true if the content attribute is set, and
|
||
* false if it is absent. On setting, the content attribute must be removed
|
||
* if the IDL attribute is set to false, and must be set to the empty
|
||
* string if the IDL attribute is set to true. (This corresponds to the
|
||
* rules for boolean content attributes.)"
|
||
*/
|
||
"boolean": {
|
||
"jsType": "boolean",
|
||
"defaultVal": false,
|
||
"domTests": ["", " foo ", undefined, null, 7, 1.5, "5%", "+100", ".5", true, false,
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{"toString":function(){return "test-toString";}},
|
||
{"valueOf":function(){return "test-valueOf";}, toString:null}],
|
||
"domExpected": function(val) {
|
||
return true;
|
||
}
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a signed integer type (long) then, on
|
||
* getting, the content attribute must be parsed according to the rules for
|
||
* parsing signed integers, and if that is successful, and the value is in
|
||
* the range of the IDL attribute's type, the resulting value must be
|
||
* returned. If, on the other hand, it fails or returns an out of range
|
||
* value, or if the attribute is absent, then the default value must be
|
||
* returned instead, or 0 if there is no default value. On setting, the
|
||
* given value must be converted to the shortest possible string
|
||
* representing the number as a valid integer and then that string must be
|
||
* used as the new content attribute value."
|
||
*/
|
||
"long": {
|
||
"jsType": "number",
|
||
"defaultVal": 0,
|
||
"domTests": [-36, -1, 0, 1, maxInt, minInt, maxInt + 1, minInt - 1,
|
||
maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
" " + binaryString + " foo ",
|
||
// Test various different whitespace. Only 20, 9, A, C,
|
||
// and D are whitespace.
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
undefined, 1.5, "5%", "+100", ".5", true, false, {"test": 6}, NaN, +Infinity,
|
||
-Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function(val) {
|
||
var parsed = ReflectionTests.parseInt(String(val));
|
||
if (parsed === false || parsed > maxInt || parsed < minInt) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [-36, -1, 0, 1, 2147483647, -2147483648],
|
||
"idlDomExpected": [-36, -1, 0, 1, 2147483647, -2147483648]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a signed integer type (long) that is
|
||
* limited to only non-negative numbers then, on getting, the content
|
||
* attribute must be parsed according to the rules for parsing non-negative
|
||
* integers, and if that is successful, and the value is in the range of
|
||
* the IDL attribute's type, the resulting value must be returned. If, on
|
||
* the other hand, it fails or returns an out of range value, or if the
|
||
* attribute is absent, the default value must be returned instead, or −1
|
||
* if there is no default value. On setting, if the value is negative, the
|
||
* user agent must fire an INDEX_SIZE_ERR exception. Otherwise, the given
|
||
* value must be converted to the shortest possible string representing the
|
||
* number as a valid non-negative integer and then that string must be used
|
||
* as the new content attribute value."
|
||
*/
|
||
"limited long": {
|
||
"jsType": "number",
|
||
"defaultVal": -1,
|
||
"domTests": [minInt - 1, minInt, -36, -1, -0, 0, 1, maxInt, maxInt + 1,
|
||
maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
" " + binaryString + " foo ",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
undefined, 1.5, "5%", "+100", ".5", true, false, {"test": 6}, NaN, +Infinity,
|
||
-Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function(val) {
|
||
var parsed = ReflectionTests.parseNonneg(String(val));
|
||
if (parsed === false || parsed > maxInt || parsed < minInt) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [minInt, -36, -1, 0, 1, maxInt],
|
||
"idlDomExpected": [null/*exception*/, null/*exception*/, null/*exception*/, 0, 1, maxInt]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is an unsigned integer type (unsigned
|
||
* long) then, on getting, the content attribute must be parsed according
|
||
* to the rules for parsing non-negative integers, and if that is
|
||
* successful, and the value is in the range 0 to 2147483647 inclusive, the
|
||
* resulting value must be returned. If, on the other hand, it fails or
|
||
* returns an out of range value, or if the attribute is absent, the
|
||
* default value must be returned instead, or 0 if there is no default
|
||
* value. On setting, the given value must be converted to the shortest
|
||
* possible string representing the number as a valid non-negative integer
|
||
* and then that string must be used as the new content attribute value."
|
||
*/
|
||
"unsigned long": {
|
||
"jsType": "number",
|
||
"defaultVal": 0,
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, 257, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function(val) {
|
||
var parsed = ReflectionTests.parseNonneg(String(val));
|
||
// Note maxInt, not maxUnsigned.
|
||
if (parsed === false || parsed < 0 || parsed > maxInt) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [0, 1, 257, maxInt, "-0", maxInt + 1, maxUnsigned],
|
||
"idlIdlExpected": [0, 1, 257, maxInt, 0, null, null],
|
||
"idlDomExpected": [0, 1, 257, maxInt, 0, null, null],
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is an unsigned integer type (unsigned
|
||
* long) that is limited to only non-negative numbers greater than zero,
|
||
* then the behavior is similar to the previous case, but zero is not
|
||
* allowed. On getting, the content attribute must first be parsed
|
||
* according to the rules for parsing non-negative integers, and if that is
|
||
* successful, and the value is in the range 1 to 2147483647 inclusive, the
|
||
* resulting value must be returned. If, on the other hand, it fails or
|
||
* returns an out of range value, or if the attribute is absent, the
|
||
* default value must be returned instead, or 1 if there is no default
|
||
* value. On setting, if the value is zero, the user agent must fire an
|
||
* INDEX_SIZE_ERR exception. Otherwise, the given value must be converted
|
||
* to the shortest possible string representing the number as a valid
|
||
* non-negative integer and then that string must be used as the new
|
||
* content attribute value."
|
||
*/
|
||
"limited unsigned long": {
|
||
"jsType": "number",
|
||
"defaultVal": 1,
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function(val) {
|
||
var parsed = ReflectionTests.parseNonneg(String(val));
|
||
// Note maxInt, not maxUnsigned.
|
||
if (parsed === false || parsed < 1 || parsed > maxInt) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [0, 1, maxInt, maxInt + 1, maxUnsigned],
|
||
"idlDomExpected": [null/*exception*/, 1, maxInt, null, null]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute has an unsigned integer type (unsigned
|
||
* long) that is limited to only non-negative numbers greater than zero
|
||
* with fallback, then the behaviour is similar to the previous case, but
|
||
* disallowed values are converted to the default value. On getting, the
|
||
* content attribute must first be parsed according to the rules for
|
||
* parsing non-negative integers, and if that is successful, and the value
|
||
* is in the range 1 to 2147483647 inclusive, the resulting value must be
|
||
* returned. If, on the other hand, it fails or returns an out of range
|
||
* value, or if the attribute is absent, the default value must be returned
|
||
* instead. On setting, first, if the new value is in the range 1 to
|
||
* 2147483647, then let n be the new value, otherwise let n be the default
|
||
* value; then, n must be converted to the shortest possible string
|
||
* representing the number as a valid non-negative integer and that string
|
||
* must be used as the new content attribute value."
|
||
*/
|
||
"limited unsigned long with fallback": {
|
||
"jsType": "number",
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function(val) {
|
||
var parsed = ReflectionTests.parseNonneg(String(val));
|
||
// Note maxInt, not maxUnsigned.
|
||
if (parsed === false || parsed < 1 || parsed > maxInt) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [0, 1, maxInt, maxInt + 1, maxUnsigned],
|
||
"idlDomExpected": [null, 1, maxInt, null, null]
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute has an unsigned integer type (unsigned
|
||
* long) that is clamped to the range [min, max], then on getting, the
|
||
* content attribute must first be parsed according to the rules for
|
||
* parsing non-negative integers, and if that is successful, and the value
|
||
* is between min and max inclusive, the resulting value must be returned.
|
||
* If it fails, the default value must be returned. If it succeeds but the
|
||
* value is less than min, min must be returned. If it succeeds but the
|
||
* value is greater than max, max must be returned. On setting, it behaves
|
||
* the same as a regular reflected unsigned integer."
|
||
*
|
||
* The data object passed to reflects must contain the keys defaultVal,
|
||
* min, and max. As with enum, domExpected is generated later once we have
|
||
* access to the min and max.
|
||
*/
|
||
"clamped unsigned long": {
|
||
"jsType": "number",
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+", "-1", "-0", "0", "1",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"idlTests": [0, 1, 257, maxInt, "-0", maxInt + 1, maxUnsigned],
|
||
"idlDomExpected": [0, 1, 257, maxInt, 0, null, null],
|
||
},
|
||
/**
|
||
* "If a reflecting IDL attribute is a floating point number type (double),
|
||
* then, on getting, the content attribute must be parsed according to the
|
||
* rules for parsing floating point number values, and if that is
|
||
* successful, the resulting value must be returned. If, on the other hand,
|
||
* it fails, or if the attribute is absent, the default value must be
|
||
* returned instead, or 0.0 if there is no default value. On setting, the
|
||
* given value must be converted to the best representation of the number
|
||
* as a floating point number and then that string must be used as the new
|
||
* content attribute value."
|
||
*
|
||
* TODO: Check this:
|
||
*
|
||
* "Except where otherwise specified, if an IDL attribute that is a
|
||
* floating point number type (double) is assigned an Infinity or
|
||
* Not-a-Number (NaN) value, a NOT_SUPPORTED_ERR exception must be raised."
|
||
*/
|
||
"double": {
|
||
"jsType": "number",
|
||
"defaultVal": 0.0,
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
"1.", "1e2", "1e+2", "1e-2", "1E2", "1E+2", "1E-2", "1.e2", "1.0e2",
|
||
"1. 1", "1 .1", "1. e2", "1 .e2", "1 e2", "1e 2", "1e -2", "1e- 2",
|
||
"1.8e308", "-1.8e308",
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function (val) {
|
||
var parsed = ReflectionTests.parseFloat(String(val));
|
||
if (parsed === false) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [ -10000000000, -1, -0, 0, 1, 10000000000,
|
||
1e-10, 1e-4, 1.5, 1e25 ],
|
||
"idlIdlExpected": function (val) {
|
||
// This is a bit heavy-weight but hopefully will give values
|
||
// that compare "better" (without introducing some tolerance)
|
||
// when the test cases are expanded with more values.
|
||
return ReflectionTests.parseFloat(String(val));
|
||
}
|
||
},
|
||
/**
|
||
* Reflected IDL attribute of type double, limited to only positive values,
|
||
* are similar to the previous case with the following exceptions:
|
||
*
|
||
* - on getting, if the parsed value is not greater than 0, then return
|
||
* the default value
|
||
* - on setting, if the value is not greater than 0, then return (leaving)
|
||
* the attribute to its previous value.
|
||
*/
|
||
"limited double": {
|
||
"jsType": "number",
|
||
"defaultVal": 0.0,
|
||
"domTests": [minInt - 1, minInt, -36, -1, 0, 1, maxInt,
|
||
maxInt + 1, maxUnsigned, maxUnsigned + 1, "", "-", "+",
|
||
"\u00097", "\u000B7", "\u000C7", "\u00207", "\u00A07", "\uFEFF7",
|
||
"\u000A7", "\u000D7", "\u20287", "\u20297", "\u16807", "\u180E7",
|
||
"\u20007", "\u20017", "\u20027", "\u20037", "\u20047", "\u20057",
|
||
"\u20067", "\u20077", "\u20087", "\u20097", "\u200A7", "\u202F7", "\u30007",
|
||
"\t\u000B7", "\n\u000B7","\f\u000B7", "\r\u000B7", "\x20\u000B7", "7\u000B",
|
||
" " + binaryString + " foo ", undefined, 1.5, "5%", "+100", ".5", true, false,
|
||
"1.", "1e2", "1e+2", "1e-2", "1E2", "1E+2", "1E-2", "1.e2", "1.0e2",
|
||
"1. 1", "1 .1", "1. e2", "1 .e2", "1 e2", "1e 2", "1e -2", "1e- 2",
|
||
"1.8e308", "-1.8e308",
|
||
{"test": 6}, NaN, +Infinity, -Infinity, "\0",
|
||
{toString:function() {return 2;}, valueOf: null},
|
||
{valueOf:function() {return 3;}, toString: null}],
|
||
"domExpected": function (val) {
|
||
var parsed = ReflectionTests.parseFloat(String(val));
|
||
if (parsed === false || parsed <= 0) {
|
||
return null;
|
||
}
|
||
return parsed;
|
||
},
|
||
"idlTests": [ -10000000000, -1, -0, 0, 1, 10000000000,
|
||
1e-10, 1e-4, 1.5, 1e25 ],
|
||
"idlIdlExpected": function (val) {
|
||
// Non-positive values are special-cased below, as they
|
||
// should be ignored, leaving the current value unchanged
|
||
|
||
// This is a bit heavy-weight but hopefully will give values
|
||
// that compare "better" (without introducing some tolerance)
|
||
// when the test cases are expanded with more values.
|
||
return ReflectionTests.parseFloat(String(val));
|
||
}
|
||
}
|
||
};
|
||
|
||
for (var type in ReflectionTests.typeMap) {
|
||
var props = ReflectionTests.typeMap[type];
|
||
var cast = window[props.jsType[0].toUpperCase() + props.jsType.slice(1)];
|
||
if (props.domExpected === undefined) {
|
||
props.domExpected = props.domTests.map(cast);
|
||
} else if (typeof props.domExpected == "function") {
|
||
props.domExpected = props.domTests.map(props.domExpected);
|
||
}
|
||
if (props.idlTests === undefined) {
|
||
props.idlTests = props.domTests;
|
||
}
|
||
if (props.idlDomExpected === undefined) {
|
||
props.idlDomExpected = props.idlTests.map(cast);
|
||
} else if (typeof props.idlDomExpected == "function") {
|
||
props.idlDomExpected = props.idlTests.map(props.idlDomExpected);
|
||
}
|
||
if (props.idlIdlExpected === undefined) {
|
||
props.idlIdlExpected = props.idlDomExpected;
|
||
} else if (typeof props.idlIdlExpected == "function") {
|
||
props.idlIdlExpected = props.idlTests.map(props.idlIdlExpected);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Tests that the JavaScript attribute named idlName on the object idlObj
|
||
* reflects the DOM attribute named domName on domObj. The data argument is an
|
||
* object that must contain at least one key, "type", which contains the
|
||
* expected type of the IDL attribute ("string", "enum", etc.). The "comment"
|
||
* key will add a parenthesized comment in the type info if there's a test
|
||
* failure, to indicate that there's something special about the element you're
|
||
* testing (like it has an attribute set to some value). Other keys in the
|
||
* data object are type-specific, e.g., "defaultVal" for numeric types. If the
|
||
* data object is a string, it's converted to {"type": data}. If idlObj is a
|
||
* string, we set idlObj = domObj = document.createElement(idlObj).
|
||
*/
|
||
ReflectionTests.reflects = function(data, idlName, idlObj, domName, domObj) {
|
||
// Do some setup first so that getTypeDescription() works in testWrapper()
|
||
if (typeof data == "string") {
|
||
data = {type: data};
|
||
}
|
||
if (domName === undefined) {
|
||
domName = idlName;
|
||
}
|
||
if (typeof idlObj == "string") {
|
||
idlObj = document.createElement(idlObj);
|
||
}
|
||
if (domObj === undefined) {
|
||
domObj = idlObj;
|
||
}
|
||
|
||
// Note: probably a hack? This kind of assumes that the variables here
|
||
// won't change over the course of the tests, which is wrong, but it's
|
||
// probably safe enough. Just don't read stuff that will change.
|
||
ReflectionHarness.currentTestInfo = {data: data, idlName: idlName, idlObj: idlObj, domName: domName, domObj: domObj};
|
||
|
||
// If we don't recognize the type, testing is impossible.
|
||
if (this.typeMap[data.type] === undefined) {
|
||
if (unimplemented.indexOf(data.type) == -1) {
|
||
unimplemented.push(data.type);
|
||
}
|
||
return;
|
||
}
|
||
|
||
var typeInfo = this.typeMap[data.type];
|
||
|
||
if (typeof data.isNullable == "undefined") {
|
||
data.isNullable = false;
|
||
}
|
||
|
||
// Test that typeof idlObj[idlName] is correct. If not, further tests are
|
||
// probably pointless, so bail out if we're not running conformance tests.
|
||
var expectedType = data.isNullable && data.defaultVal === null ? "object"
|
||
: typeInfo.jsType;
|
||
ReflectionHarness.test(function() {
|
||
ReflectionHarness.assertEquals(typeof idlObj[idlName], expectedType);
|
||
}, "typeof IDL attribute");
|
||
|
||
if (!ReflectionHarness.conformanceTesting &&
|
||
typeof idlObj[idlName] !== expectedType) {
|
||
return;
|
||
}
|
||
|
||
// Test default
|
||
var defaultVal = data.defaultVal;
|
||
if (defaultVal === undefined) {
|
||
defaultVal = typeInfo.defaultVal;
|
||
}
|
||
if ((domObj.localName === "form" && domName === "action") ||
|
||
(["button", "input"].includes(domObj.localName) &&
|
||
domName === "formAction")) {
|
||
// Hard-coded special case
|
||
defaultVal = domObj.ownerDocument.URL;
|
||
}
|
||
if (!data.customGetter && (defaultVal !== null || data.isNullable)) {
|
||
ReflectionHarness.test(function() {
|
||
ReflectionHarness.assertEquals(idlObj[idlName], defaultVal);
|
||
}, "IDL get with DOM attribute unset");
|
||
}
|
||
|
||
var domTests = typeInfo.domTests.slice(0);
|
||
var domExpected = typeInfo.domExpected.map(function(val) { return val === null ? defaultVal : val; });
|
||
var idlTests = typeInfo.idlTests.slice(0);
|
||
var idlDomExpected = typeInfo.idlDomExpected.map(function(val) { return val === null ? defaultVal : val; });
|
||
var idlIdlExpected = typeInfo.idlIdlExpected.map(function(val) { return val === null ? defaultVal : val; });
|
||
switch (data.type) {
|
||
// Extra tests and other special-casing
|
||
case "boolean":
|
||
domTests.push(domName);
|
||
domExpected.push(true);
|
||
break;
|
||
|
||
case "enum":
|
||
// Whee, enum is complicated.
|
||
if (typeof data.invalidVal == "undefined") {
|
||
data.invalidVal = defaultVal;
|
||
}
|
||
if (typeof data.nonCanon == "undefined") {
|
||
data.nonCanon = {};
|
||
}
|
||
for (var i = 0; i < data.keywords.length; i++) {
|
||
if (data.keywords[i] != "") {
|
||
domTests.push(data.keywords[i], "x" + data.keywords[i], data.keywords[i] + "\0");
|
||
idlTests.push(data.keywords[i], "x" + data.keywords[i], data.keywords[i] + "\0");
|
||
}
|
||
|
||
if (data.keywords[i].length > 1) {
|
||
var sliced = data.keywords[i].slice(1);
|
||
// If slicing a value yields another valid value, then skip it since it results in duplicate tests.
|
||
if (data.keywords.indexOf(sliced) == -1) {
|
||
domTests.push(sliced);
|
||
idlTests.push(sliced);
|
||
}
|
||
}
|
||
|
||
if (data.keywords[i] != data.keywords[i].toLowerCase()) {
|
||
domTests.push(data.keywords[i].toLowerCase());
|
||
idlTests.push(data.keywords[i].toLowerCase());
|
||
}
|
||
if (data.keywords[i] != data.keywords[i].toUpperCase()) {
|
||
domTests.push(data.keywords[i].toUpperCase());
|
||
idlTests.push(data.keywords[i].toUpperCase());
|
||
}
|
||
if (data.keywords[i].indexOf("k") != -1) {
|
||
domTests.push(data.keywords[i].replace(/k/g, "\u212A"));
|
||
idlTests.push(data.keywords[i].replace(/k/g, "\u212A"));
|
||
}
|
||
if (data.keywords[i].indexOf("s") != -1) {
|
||
domTests.push(data.keywords[i].replace(/s/g, "\u017F"));
|
||
idlTests.push(data.keywords[i].replace(/s/g, "\u017F"));
|
||
}
|
||
}
|
||
|
||
// Per spec, the expected DOM values are the same as the value we set
|
||
// it to.
|
||
if (!data.isNullable) {
|
||
idlDomExpected = idlTests.slice(0);
|
||
} else {
|
||
idlDomExpected = [];
|
||
for (var i = 0; i < idlTests.length; i++) {
|
||
idlDomExpected.push((idlTests[i] === null || idlTests[i] === undefined) ? null : idlTests[i]);
|
||
}
|
||
}
|
||
|
||
// Now we have the fun of calculating what the expected IDL values are.
|
||
domExpected = [];
|
||
idlIdlExpected = [];
|
||
for (var i = 0; i < domTests.length; i++) {
|
||
domExpected.push(this.enumExpected(data.keywords, data.nonCanon, data.invalidVal, domTests[i]));
|
||
}
|
||
for (var i = 0; i < idlTests.length; i++) {
|
||
if (data.isNullable && (idlTests[i] === null || idlTests[i] === undefined)) {
|
||
idlIdlExpected.push(null);
|
||
} else {
|
||
idlIdlExpected.push(this.enumExpected(data.keywords, data.nonCanon, data.invalidVal, idlTests[i]));
|
||
}
|
||
}
|
||
break;
|
||
|
||
case "string":
|
||
if ("treatNullAsEmptyString" in data) {
|
||
for (var i = 0; i < idlTests.length; i++) {
|
||
if (idlTests[i] === null) {
|
||
idlDomExpected[i] = idlIdlExpected[i] = "";
|
||
}
|
||
}
|
||
}
|
||
break;
|
||
|
||
case "clamped unsigned long":
|
||
[data.min - 1, data.min, data.max, data.max + 1].forEach(function(val) {
|
||
if (domTests.indexOf(val) == -1) {
|
||
domTests.push(val);
|
||
}
|
||
if (idlTests.indexOf(val) == -1 && 0 <= val && val <= maxUnsigned) {
|
||
idlTests.push(val);
|
||
if (typeof val != "number") {
|
||
val = ReflectionTests.parseNonneg(val);
|
||
}
|
||
idlDomExpected.push(val > maxInt ? null : val);
|
||
}
|
||
});
|
||
|
||
// Rewrite expected values
|
||
domExpected = domTests.map(function(val) {
|
||
var parsed = ReflectionTests.parseNonneg(String(val));
|
||
if (parsed === false) {
|
||
return defaultVal;
|
||
}
|
||
if (parsed < data.min) {
|
||
return data.min;
|
||
}
|
||
if (parsed > data.max) {
|
||
return data.max;
|
||
}
|
||
return parsed;
|
||
});
|
||
idlIdlExpected = idlTests.map(function(val) {
|
||
if (typeof val != "number") {
|
||
val = ReflectionTests.parseNonneg(val);
|
||
}
|
||
if (val < 0 || val > maxUnsigned) {
|
||
throw "Test bug: val should be an unsigned long";
|
||
}
|
||
if (val > maxInt) {
|
||
return defaultVal;
|
||
}
|
||
if (val < data.min) {
|
||
return data.min;
|
||
}
|
||
if (val > data.max) {
|
||
return data.max;
|
||
}
|
||
return val;
|
||
});
|
||
break;
|
||
}
|
||
if (domObj.tagName.toLowerCase() == "canvas" && (domName == "width" || domName == "height")) {
|
||
// Opera tries to allocate a canvas with the given width and height, so
|
||
// it OOMs when given excessive sizes. This is permissible under the
|
||
// hardware-limitations clause, so cut out those checks. TODO: Must be
|
||
// a way to make this more succinct.
|
||
domTests = domTests.filter(function(element, index, array) { return domExpected[index] < 1000; });
|
||
domExpected = domExpected.filter(function(element, index, array) { return element < 1000; });
|
||
idlTests = idlTests.filter(function(element, index, array) { return idlIdlExpected[index] < 1000; });
|
||
idlDomExpected = idlDomExpected.filter(function(element, index, array) { return idlIdlExpected[index] < 1000; });
|
||
idlIdlExpected = idlIdlExpected.filter(function(element, index, array) { return idlIdlExpected[index] < 1000; });
|
||
}
|
||
if ((domObj.localName === "form" && domName === "action") ||
|
||
(["button", "input"].includes(domObj.localName) &&
|
||
domName === "formAction")) {
|
||
// Hard-coded special case
|
||
for (var i = 0; i < domTests.length; i++) {
|
||
if (domTests[i] === "") {
|
||
domExpected[i] = domObj.ownerDocument.URL;
|
||
}
|
||
}
|
||
for (var i = 0; i < idlTests.length; i++) {
|
||
if (idlTests[i] === "") {
|
||
idlIdlExpected[i] = domObj.ownerDocument.URL;
|
||
}
|
||
}
|
||
}
|
||
if (data.customGetter) {
|
||
// These are reflected only on setting, not getting
|
||
domTests = [];
|
||
domExpected = [];
|
||
idlIdlExpected = idlIdlExpected.map(() => null);
|
||
}
|
||
|
||
for (var i = 0; i < domTests.length; i++) {
|
||
if (domExpected[i] === null && !data.isNullable) {
|
||
// If you follow all the complicated logic here, you'll find that
|
||
// this will only happen if there's no expected value at all (like
|
||
// for tabIndex, where the default is too complicated). So skip
|
||
// the test.
|
||
continue;
|
||
}
|
||
ReflectionHarness.test(function() {
|
||
domObj.setAttribute(domName, domTests[i]);
|
||
ReflectionHarness.assertEquals(domObj.getAttribute(domName),
|
||
String(domTests[i]), "getAttribute()");
|
||
ReflectionHarness.assertEquals(idlObj[idlName], domExpected[i],
|
||
"IDL get");
|
||
}, "setAttribute() to " + ReflectionHarness.stringRep(domTests[i]));
|
||
}
|
||
|
||
for (var i = 0; i < idlTests.length; i++) {
|
||
ReflectionHarness.test(function() {
|
||
if ((data.type == "limited long" && idlTests[i] < 0) ||
|
||
(data.type == "limited unsigned long" && idlTests[i] == 0)) {
|
||
ReflectionHarness.assertThrows("IndexSizeError", function() {
|
||
idlObj[idlName] = idlTests[i];
|
||
});
|
||
} else if (data.type == "limited double" && idlTests[i] <= 0) {
|
||
domObj.setAttribute(domName, "previous value");
|
||
var previousIdl = idlObj[idlName]; // should be the default value
|
||
idlObj[idlName] = idlTests[i];
|
||
ReflectionHarness.assertEquals(domObj.getAttribute(domName),
|
||
"previous value", "getAttribute()");
|
||
ReflectionHarness.assertEquals(idlObj[idlName], previousIdl, "IDL get");
|
||
} else {
|
||
var previousValue = domObj.getAttribute(domName);
|
||
idlObj[idlName] = idlTests[i];
|
||
if (data.type == "boolean") {
|
||
// Special case yay
|
||
ReflectionHarness.assertEquals(domObj.hasAttribute(domName),
|
||
Boolean(idlTests[i]), "hasAttribute()");
|
||
} else if (idlDomExpected[i] !== null || data.isNullable) {
|
||
var expected = idlDomExpected[i] + "";
|
||
if (data.isNullable && idlDomExpected[i] === null) {
|
||
expected = null;
|
||
} else if (idlName == "nonce") {
|
||
// nonce doesn't reflect the value, as per /content-security-policy/nonce-hiding/
|
||
// tests that confirm that retrieving the nonce value post IDL change does not
|
||
// reflect back to the attribute (for security reasons)
|
||
expected = previousValue;
|
||
}
|
||
ReflectionHarness.assertEquals(domObj.getAttribute(domName), expected,
|
||
"getAttribute()");
|
||
}
|
||
if (idlIdlExpected[i] !== null || data.isNullable) {
|
||
ReflectionHarness.assertEquals(idlObj[idlName], idlIdlExpected[i], "IDL get");
|
||
}
|
||
}
|
||
}, "IDL set to " + ReflectionHarness.stringRep(idlTests[i]));
|
||
}
|
||
};
|
||
|
||
function toASCIILowerCase(str) {
|
||
return str.replace(/[A-Z]/g, function(m) { return m.toLowerCase(); });
|
||
}
|
||
|
||
/**
|
||
* If we have an enumerated attribute limited to the array of values in
|
||
* keywords, with nonCanon being a map of non-canonical values to their
|
||
* canonical equivalents, and invalidVal being the invalid value default (or ""
|
||
* for none), then what would we expect from an IDL get if the content
|
||
* attribute is equal to contentVal?
|
||
*/
|
||
ReflectionTests.enumExpected = function(keywords, nonCanon, invalidVal, contentVal) {
|
||
var ret = invalidVal;
|
||
for (var i = 0; i < keywords.length; i++) {
|
||
if (toASCIILowerCase(String(contentVal)) === toASCIILowerCase(keywords[i])) {
|
||
ret = keywords[i];
|
||
break;
|
||
}
|
||
}
|
||
if (typeof nonCanon[ret] != "undefined") {
|
||
return nonCanon[ret];
|
||
}
|
||
return ret;
|
||
};
|
||
|
||
/**
|
||
* Now we have the data structures that tell us which elements have which
|
||
* attributes.
|
||
*
|
||
* The elements object (which must have been defined in earlier files) is a map
|
||
* from element name to an object whose keys are IDL attribute names and whose
|
||
* values are types. A type is of the same format as
|
||
* ReflectionTests.reflects() accepts, except that there's an extra optional
|
||
* domAttrName key that gets passed as the fourth argument to reflects() if
|
||
* it's provided. (TODO: drop the fourth and fifth reflects() arguments and
|
||
* make it take them from the dictionary instead?)
|
||
*/
|
||
|
||
// Now we actually run all the tests.
|
||
var unimplemented = [];
|
||
for (var element in elements) {
|
||
ReflectionTests.reflects("string", "title", element);
|
||
ReflectionTests.reflects("string", "lang", element);
|
||
ReflectionTests.reflects({type: "enum", keywords: ["ltr", "rtl", "auto"]}, "dir", element);
|
||
ReflectionTests.reflects("string", "className", element, "class");
|
||
ReflectionTests.reflects("tokenlist", "classList", element, "class");
|
||
ReflectionTests.reflects("boolean", "autofocus", element);
|
||
ReflectionTests.reflects("boolean", "hidden", element);
|
||
ReflectionTests.reflects("string", "accessKey", element);
|
||
// Don't try to test the defaultVal -- it should be either 0 or -1, but the
|
||
// rules are complicated, and a lot of them are SHOULDs.
|
||
ReflectionTests.reflects({type: "long", defaultVal: null}, "tabIndex", element);
|
||
// TODO: classList, contextMenu, itemProp, itemRef
|
||
|
||
for (var idlAttrName in elements[element]) {
|
||
var type = elements[element][idlAttrName];
|
||
ReflectionTests.reflects(type, idlAttrName, element,
|
||
typeof type == "object" && "domAttrName" in type ? type.domAttrName : idlAttrName);
|
||
}
|
||
}
|
||
|
||
for (var i = 0; i < extraTests.length; i++) {
|
||
extraTests[i]();
|
||
}
|
||
|
||
var time = document.getElementById("time");
|
||
if (time) {
|
||
time.innerHTML = (new Date().getTime() - ReflectionTests.start)/1000;
|
||
}
|
||
|
||
if (unimplemented.length) {
|
||
var p = document.createElement("p");
|
||
p.textContent = "(Note: missing tests for types " + unimplemented.join(", ") + ".)";
|
||
document.body.appendChild(p);
|
||
}
|