2012-08-27 05:48:49 +02:00
|
|
|
// Copyright 2012 Mozilla Corporation. All rights reserved.
|
|
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @description Tests that obj meets the requirements for built-in objects
|
|
|
|
* defined by the introduction of chapter 15 of the ECMAScript Language Specification.
|
|
|
|
* @param {Object} obj the object to be tested.
|
|
|
|
* @param {boolean} isFunction whether the specification describes obj as a function.
|
|
|
|
* @param {boolean} isConstructor whether the specification describes obj as a constructor.
|
|
|
|
* @param {String[]} properties an array with the names of the built-in properties of obj,
|
|
|
|
* excluding length, prototype, or properties with non-default attributes.
|
|
|
|
* @param {number} length for functions only: the length specified for the function
|
|
|
|
* or derived from the argument list.
|
|
|
|
* @author Norbert Lindenberg
|
|
|
|
*/
|
|
|
|
|
|
|
|
function testBuiltInObject(obj, isFunction, isConstructor, properties, length) {
|
|
|
|
|
|
|
|
if (obj === undefined) {
|
|
|
|
$ERROR("Object being tested is undefined.");
|
|
|
|
}
|
|
|
|
|
|
|
|
var objString = Object.prototype.toString.call(obj);
|
|
|
|
if (isFunction) {
|
|
|
|
if (objString !== "[object Function]") {
|
|
|
|
$ERROR("The [[Class]] internal property of a built-in function must be " +
|
|
|
|
"\"Function\", but toString() returns " + objString);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (objString !== "[object Object]") {
|
|
|
|
$ERROR("The [[Class]] internal property of a built-in non-function object must be " +
|
|
|
|
"\"Object\", but toString() returns " + objString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Object.isExtensible(obj)) {
|
|
|
|
$ERROR("Built-in objects must be extensible.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isFunction && Object.getPrototypeOf(obj) !== Function.prototype) {
|
|
|
|
$ERROR("Built-in functions must have Function.prototype as their prototype.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isConstructor && Object.getPrototypeOf(obj.prototype) !== Object.prototype) {
|
|
|
|
$ERROR("Built-in prototype objects must have Object.prototype as their prototype.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// verification of the absence of the [[Construct]] internal property has
|
|
|
|
// been moved to the end of the test
|
|
|
|
|
|
|
|
// verification of the absence of the prototype property has
|
|
|
|
// been moved to the end of the test
|
|
|
|
|
|
|
|
if (isFunction) {
|
|
|
|
|
|
|
|
if (typeof obj.length !== "number" || obj.length !== Math.floor(obj.length)) {
|
|
|
|
$ERROR("Built-in functions must have a length property with an integer value.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj.length !== length) {
|
2012-09-10 20:53:56 +02:00
|
|
|
$ERROR("Function's length property doesn't have specified value; expected " +
|
|
|
|
length + ", got " + obj.length + ".");
|
2012-08-27 05:48:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var desc = Object.getOwnPropertyDescriptor(obj, "length");
|
|
|
|
if (desc.writable) {
|
|
|
|
$ERROR("The length property of a built-in function must not be writable.");
|
|
|
|
}
|
|
|
|
if (desc.enumerable) {
|
|
|
|
$ERROR("The length property of a built-in function must not be enumerable.");
|
|
|
|
}
|
2015-03-11 15:25:56 +01:00
|
|
|
if (!desc.configurable) {
|
|
|
|
$ERROR("The length property of a built-in function must be configurable.");
|
2012-08-27 05:48:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
properties.forEach(function(prop) {
|
|
|
|
var desc = Object.getOwnPropertyDescriptor(obj, prop);
|
|
|
|
if (desc === undefined) {
|
|
|
|
$ERROR("Missing property " + prop + ".");
|
|
|
|
}
|
|
|
|
// accessor properties don't have writable attribute
|
|
|
|
if (desc.hasOwnProperty("writable") && !desc.writable) {
|
2015-04-21 23:58:10 +02:00
|
|
|
$ERROR("The " + prop + " property of this built-in object must be writable.");
|
2012-08-27 05:48:49 +02:00
|
|
|
}
|
|
|
|
if (desc.enumerable) {
|
2015-04-21 23:58:10 +02:00
|
|
|
$ERROR("The " + prop + " property of this built-in object must not be enumerable.");
|
2012-08-27 05:48:49 +02:00
|
|
|
}
|
|
|
|
if (!desc.configurable) {
|
2015-04-21 23:58:10 +02:00
|
|
|
$ERROR("The " + prop + " property of this built-in object must be configurable.");
|
2012-08-27 05:48:49 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// The remaining sections have been moved to the end of the test because
|
|
|
|
// unbound non-constructor functions written in JavaScript cannot possibly
|
|
|
|
// pass them, and we still want to test JavaScript implementations as much
|
|
|
|
// as possible.
|
|
|
|
|
|
|
|
var exception;
|
|
|
|
if (isFunction && !isConstructor) {
|
|
|
|
// this is not a complete test for the presence of [[Construct]]:
|
|
|
|
// if it's absent, the exception must be thrown, but it may also
|
|
|
|
// be thrown if it's present and just has preconditions related to
|
|
|
|
// arguments or the this value that this statement doesn't meet.
|
|
|
|
try {
|
|
|
|
/*jshint newcap:false*/
|
|
|
|
var instance = new obj();
|
|
|
|
} catch (e) {
|
|
|
|
exception = e;
|
|
|
|
}
|
|
|
|
if (exception === undefined || exception.name !== "TypeError") {
|
|
|
|
$ERROR("Built-in functions that aren't constructors must throw TypeError when " +
|
|
|
|
"used in a \"new\" statement.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isFunction && !isConstructor && obj.hasOwnProperty("prototype")) {
|
|
|
|
$ERROR("Built-in functions that aren't constructors must not have a prototype property.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// passed the complete test!
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|