| Index: test/mjsunit/double-equals.js
|
| diff --git a/test/mjsunit/double-equals.js b/test/mjsunit/double-equals.js
|
| index a68d7eaf9835b1962ba89d9d6a01dfb5ac24ab22..5ebf92ca7bd4bb31df070555a59ad8b34953775e 100644
|
| --- a/test/mjsunit/double-equals.js
|
| +++ b/test/mjsunit/double-equals.js
|
| @@ -31,84 +31,206 @@
|
| * implementation of assertEquals.
|
| */
|
|
|
| -assertTrue (void 0 == void 0, "void 0 == void 0");
|
| -assertTrue (null == null, "null == null");
|
| -assertFalse(NaN == NaN, "NaN == NaN");
|
| -assertFalse(NaN == 0, "NaN == 0");
|
| -assertFalse(0 == NaN, "0 == NaN");
|
| -assertFalse(NaN == Infinity, "NaN == Inf");
|
| -assertFalse(Infinity == NaN, "Inf == NaN");
|
| -
|
| -assertTrue(Number.MAX_VALUE == Number.MAX_VALUE, "MAX == MAX");
|
| -assertTrue(Number.MIN_VALUE == Number.MIN_VALUE, "MIN == MIN");
|
| -assertTrue(Infinity == Infinity, "Inf == Inf");
|
| -assertTrue(-Infinity == -Infinity, "-Inf == -Inf");
|
| -
|
| -assertTrue(0 == 0, "0 == 0");
|
| -assertTrue(0 == -0, "0 == -0");
|
| -assertTrue(-0 == 0, "-0 == 0");
|
| -assertTrue(-0 == -0, "-0 == -0");
|
| -
|
| -assertFalse(0.9 == 1, "0.9 == 1");
|
| -assertFalse(0.999999 == 1, "0.999999 == 1");
|
| -assertFalse(0.9999999999 == 1, "0.9999999999 == 1");
|
| -assertFalse(0.9999999999999 == 1, "0.9999999999999 == 1");
|
| -
|
| -assertTrue('hello' == 'hello', "'hello' == 'hello'");
|
| -
|
| -assertTrue (true == true, "true == true");
|
| -assertTrue (false == false, "false == false");
|
| -assertFalse(true == false, "true == false");
|
| -assertFalse(false == true, "false == true");
|
| -
|
| -assertFalse(new Wrapper(null) == new Wrapper(null), "new Wrapper(null) == new Wrapper(null)");
|
| -assertFalse(new Boolean(true) == new Boolean(true), "new Boolean(true) == new Boolean(true)");
|
| -assertFalse(new Boolean(false) == new Boolean(false), "new Boolean(false) == new Boolean(false)");
|
| +function testEqual(a, b) {
|
| + assertTrue(a == b);
|
| + assertTrue(b == a);
|
| + assertFalse(a != b);
|
| + assertFalse(b != a);
|
| +}
|
| +
|
| +function testNotEqual(a, b) {
|
| + assertFalse(a == b);
|
| + assertFalse(b == a);
|
| + assertTrue(a != b);
|
| + assertTrue(b != a);
|
| +}
|
| +
|
| +// Object where ToPrimitive returns value.
|
| +function Wrapper(value) {
|
| + this.value = value;
|
| + this.valueOf = function () { return this.value; };
|
| +}
|
| +
|
| +// Object where ToPrimitive returns value by failover to toString when
|
| +// valueOf isn't a function.
|
| +function Wrapper2(value) {
|
| + this.value = value;
|
| + this.valueOf = null;
|
| + this.toString = function () { return this.value; };
|
| +}
|
| +
|
| +
|
| +// Compare values of same type.
|
| +
|
| +// Numbers are equal if same, unless NaN, which isn't equal to anything, and
|
| +// +/-0 being equal.
|
| +
|
| +testNotEqual(NaN, NaN);
|
| +testNotEqual(NaN, 0);
|
| +testNotEqual(NaN, Infinity);
|
| +
|
| +testEqual(Number.MAX_VALUE, Number.MAX_VALUE);
|
| +testEqual(Number.MIN_VALUE, Number.MIN_VALUE);
|
| +testEqual(Infinity, Infinity);
|
| +testEqual(-Infinity, -Infinity);
|
| +
|
| +testEqual(0, 0);
|
| +testEqual(0, -0);
|
| +testEqual(-0, -0);
|
| +
|
| +testNotEqual(0.9, 1);
|
| +testNotEqual(0.999999, 1);
|
| +testNotEqual(0.9999999999, 1);
|
| +testNotEqual(0.9999999999999, 1);
|
| +
|
| +// Strings are equal if containing the same code points.
|
| +
|
| +testEqual('hello', 'hello');
|
| +testEqual('hello', 'hel' + 'lo');
|
| +testEqual('', '');
|
| +testEqual('\u0020\x20', ' '); // Escapes are not part of the value.
|
| +
|
| +// Booleans are equal if they are the same.
|
| +
|
| +testEqual(true, true);
|
| +testEqual(false, false);
|
| +testNotEqual(true, false);
|
| +
|
| +// Null and undefined are equal to themselves.
|
| +
|
| +testEqual(null, null);
|
| +testEqual(undefined, undefined);
|
| +
|
| +// Objects are equal if they are the same object only.
|
| +
|
| +testEqual(Math, Math);
|
| +testEqual(Object.prototype, Object.prototype);
|
| +
|
|
|
| (function () {
|
| var x = new Wrapper(null);
|
| var y = x, z = x;
|
| - assertTrue(y == x);
|
| + testEqual(y, x);
|
| })();
|
|
|
| (function () {
|
| var x = new Boolean(true);
|
| var y = x, z = x;
|
| - assertTrue(y == x);
|
| + testEqual(y, x);
|
| })();
|
|
|
| (function () {
|
| var x = new Boolean(false);
|
| var y = x, z = x;
|
| - assertTrue(y == x);
|
| + testEqual(y, x);
|
| })();
|
|
|
| -assertTrue(null == void 0, "null == void 0");
|
| -assertTrue(void 0 == null, "void 0 == null");
|
| -assertFalse(new Wrapper(null) == null, "new Wrapper(null) == null");
|
| -assertFalse(null == new Wrapper(null), "null == new Wrapper(null)");
|
| -
|
| -assertTrue(1 == '1', "1 == '1");
|
| -assertTrue(255 == '0xff', "255 == '0xff'");
|
| -assertTrue(0 == '\r', "0 == '\\r'");
|
| -assertTrue(1e19 == '1e19', "1e19 == '1e19'");
|
| -
|
| -assertTrue(new Boolean(true) == true, "new Boolean(true) == true");
|
| -assertTrue(new Boolean(false) == false, "new Boolean(false) == false");
|
| -assertTrue(true == new Boolean(true), "true == new Boolean(true)");
|
| -assertTrue(false == new Boolean(false), "false == new Boolean(false)");
|
| -
|
| -assertTrue(Boolean(true) == true, "Boolean(true) == true");
|
| -assertTrue(Boolean(false) == false, "Boolean(false) == false");
|
| -assertTrue(true == Boolean(true), "true == Boolean(true)");
|
| -assertTrue(false == Boolean(false), "false == Boolean(false)");
|
| -
|
| -assertTrue(new Wrapper(true) == true, "new Wrapper(true) == true");
|
| -assertTrue(new Wrapper(false) == false, "new Wrapper(false) == false");
|
| -assertTrue(true == new Wrapper(true), "true = new Wrapper(true)");
|
| -assertTrue(false == new Wrapper(false), "false = new Wrapper(false)");
|
| -
|
| -function Wrapper(value) {
|
| - this.value = value;
|
| - this.valueOf = function () { return this.value; };
|
| +// Test comparing values of different types.
|
| +
|
| +// Null and undefined are equal to each-other, and to nothing else.
|
| +testEqual(null, undefined);
|
| +testEqual(undefined, null);
|
| +
|
| +testNotEqual(null, new Wrapper(null));
|
| +testNotEqual(null, 0);
|
| +testNotEqual(null, false);
|
| +testNotEqual(null, "");
|
| +testNotEqual(null, new Object());
|
| +testNotEqual(undefined, new Wrapper(undefined));
|
| +testNotEqual(undefined, 0);
|
| +testNotEqual(undefined, false);
|
| +testNotEqual(undefined, "");
|
| +testNotEqual(undefined, new Object());
|
| +
|
| +// Numbers compared to Strings will convert the string to a number using
|
| +// the internal ToNumber conversion.
|
| +
|
| +testEqual(1, '1');
|
| +testEqual(255, '0xff');
|
| +testEqual(0, '\r'); // ToNumber ignores tailing and trailing whitespace.
|
| +testEqual(1e19, '1e19');
|
| +testEqual(Infinity, "Infinity");
|
| +
|
| +// Booleans compared to anything else will be converted to numbers.
|
| +testEqual(false, 0);
|
| +testEqual(true, 1);
|
| +testEqual(false, "0"); // String also converted to number.
|
| +testEqual(true, "1");
|
| +
|
| +// Objects compared to Number or String (or Boolean, since that's converted
|
| +// to Number too) is converted to primitive using ToPrimitive with NO HINT.
|
| +// Having no hint means Date gets a string hint, and everything else gets
|
| +// a number hint.
|
| +
|
| +testEqual(new Boolean(true), true);
|
| +testEqual(new Boolean(true), 1); // First to primtive boolean, then to number.
|
| +testEqual(new Boolean(false), false);
|
| +testEqual(new Boolean(false), 0);
|
| +
|
| +testEqual(new Wrapper(true), true);
|
| +testEqual(new Wrapper(true), 1);
|
| +testEqual(new Wrapper(false), false);
|
| +testEqual(new Wrapper(false), 0);
|
| +
|
| +testEqual(new Wrapper2(true), true);
|
| +testEqual(new Wrapper2(true), 1);
|
| +testEqual(new Wrapper2(false), false);
|
| +testEqual(new Wrapper2(false), 0);
|
| +
|
| +testEqual(new Number(1), true);
|
| +testEqual(new Number(1), 1);
|
| +testEqual(new Number(0), false);
|
| +testEqual(new Number(0), 0);
|
| +
|
| +// Date objects convert to string, not number (and the string does not
|
| +// convert to the number).
|
| +testEqual(new Date(42), String(new Date(42)));
|
| +testNotEqual(new Date(42), Number(new Date(42)));
|
| +var dnow = new Date();
|
| +testEqual(dnow, dnow);
|
| +testEqual(dnow, String(dnow));
|
| +testNotEqual(dnow, Number(dnow));
|
| +
|
| +// Doesn't just call toString, but uses ToPrimitive which tries toString first
|
| +// and valueOf second.
|
| +dnow.toString = null;
|
| +testEqual(dnow, Number(dnow));
|
| +dnow.valueOf = function () { return "42"; };
|
| +testEqual(dnow, 42);
|
| +dnow.toString = function () { return "1"; };
|
| +testEqual(dnow, true);
|
| +
|
| +
|
| +// Objects compared to other objects, or to null and undefined, are not
|
| +// converted to primitive.
|
| +testNotEqual(new Wrapper(null), new Wrapper(null));
|
| +testNotEqual(new Boolean(true), new Boolean(true));
|
| +testNotEqual(new Boolean(false), new Boolean(false));
|
| +testNotEqual(new String("a"), new String("a"));
|
| +testNotEqual(new Number(42), new Number(42));
|
| +testNotEqual(new Date(42), new Date(42));
|
| +testNotEqual(new Array(42), new Array(42));
|
| +testNotEqual(new Object(), new Object());
|
| +
|
| +// Object that can't be converted to primitive.
|
| +var badObject = {
|
| + valueOf: null,
|
| + toString: function() {
|
| + return this; // Not primitive.
|
| + }
|
| +};
|
| +
|
| +testEqual(badObject, badObject);
|
| +testNotEqual(badObject, {});
|
| +testNotEqual(badObject, null);
|
| +testNotEqual(badObject, undefined);
|
| +// Forcing conversion will throw.
|
| +function testBadConversion(value) {
|
| + assertThrows(function() { return badObject == value; });
|
| + assertThrows(function() { return badObject != value; });
|
| + assertThrows(function() { return value == badObject; });
|
| + assertThrows(function() { return value != badObject; });
|
| }
|
| +testBadConversion(0);
|
| +testBadConversion("string");
|
| +testBadConversion(true);
|
|
|