diff --git a/lib/chai/assertion.js b/lib/chai/assertion.js index 9a3846b4b..2a97d21a6 100644 --- a/lib/chai/assertion.js +++ b/lib/chai/assertion.js @@ -100,7 +100,8 @@ module.exports = function (_chai, util) { Assertion.prototype.assert = function (expr, msg, negateMsg, expected, _actual, showDiff) { var ok = util.test(this, arguments); - if (true !== showDiff) showDiff = false; + if (false !== showDiff) showDiff = true; + if (undefined === expected && undefined === _actual) showDiff = false; if (true !== config.showDiff) showDiff = false; if (!ok) { diff --git a/lib/chai/core/assertions.js b/lib/chai/core/assertions.js index d38aaaed6..ab6a9f648 100644 --- a/lib/chai/core/assertions.js +++ b/lib/chai/core/assertions.js @@ -446,7 +446,7 @@ module.exports = function (chai, _) { * ### .NaN * Asserts that the target is `NaN`. * - * expect('foo').to.be.NaN; + * expect(NaN).to.be.NaN; * expect(4).not.to.be.NaN; * * @name NaN @@ -456,7 +456,7 @@ module.exports = function (chai, _) { Assertion.addProperty('NaN', function () { this.assert( - isNaN(flag(this, 'object')) + _.isNaN(flag(this, 'object')) , 'expected #{this} to be NaN' , 'expected #{this} not to be NaN' ); @@ -507,17 +507,10 @@ module.exports = function (chai, _) { */ Assertion.addProperty('empty', function () { - var obj = flag(this, 'object') - , expected = obj; - - if (Array.isArray(obj) || 'string' === typeof object) { - expected = obj.length; - } else if (typeof obj === 'object') { - expected = Object.keys(obj).length; - } - + var obj = flag(this, 'object'); + new Assertion(obj).to.exist; this.assert( - !expected + Object.keys(Object(obj)).length === 0 , 'expected #{this} to be empty' , 'expected #{this} not to be empty' ); @@ -629,14 +622,15 @@ module.exports = function (chai, _) { /** * ### .above(value) * - * Asserts that the target is greater than `value`. + * Asserts that the number target is greater than `value`. * * expect(10).to.be.above(5); * * Can also be used in conjunction with `length` to * assert a minimum length. The benefit being a * more informative error message than if the length - * was supplied directly. + * was supplied directly. In this case the target must + * have a length property. * * expect('foo').to.have.length.above(2); * expect([ 1, 2, 3 ]).to.have.length.above(2); @@ -652,9 +646,20 @@ module.exports = function (chai, _) { function assertAbove (n, msg) { if (msg) flag(this, 'message', msg); - var obj = flag(this, 'object'); - if (flag(this, 'doLength')) { + var obj = flag(this, 'object') + , doLength = flag(this, 'doLength'); + + if (doLength) { new Assertion(obj, msg).to.have.property('length'); + } else { + new Assertion(obj, msg).is.a('number'); + } + + if (_.type(n) !== 'number') { + throw new Error('the argument to above must be a number'); + } + + if (doLength) { var len = obj.length; this.assert( len > n @@ -679,14 +684,15 @@ module.exports = function (chai, _) { /** * ### .least(value) * - * Asserts that the target is greater than or equal to `value`. + * Asserts that the number target is greater than or equal to `value`. * * expect(10).to.be.at.least(10); * * Can also be used in conjunction with `length` to * assert a minimum length. The benefit being a * more informative error message than if the length - * was supplied directly. + * was supplied directly. In this case the target must + * have a length property. * * expect('foo').to.have.length.of.at.least(2); * expect([ 1, 2, 3 ]).to.have.length.of.at.least(3); @@ -701,9 +707,20 @@ module.exports = function (chai, _) { function assertLeast (n, msg) { if (msg) flag(this, 'message', msg); - var obj = flag(this, 'object'); - if (flag(this, 'doLength')) { + var obj = flag(this, 'object') + , doLength = flag(this, 'doLength'); + + if (doLength) { new Assertion(obj, msg).to.have.property('length'); + } else { + new Assertion(obj, msg).is.a('number'); + } + + if (_.type(n) !== 'number') { + throw new Error('the argument to least must be a number'); + } + + if (doLength) { var len = obj.length; this.assert( len >= n @@ -727,14 +744,15 @@ module.exports = function (chai, _) { /** * ### .below(value) * - * Asserts that the target is less than `value`. + * Asserts that the number target is less than `value`. * * expect(5).to.be.below(10); * * Can also be used in conjunction with `length` to * assert a maximum length. The benefit being a * more informative error message than if the length - * was supplied directly. + * was supplied directly. In this case the target must + * have a length property. * * expect('foo').to.have.length.below(4); * expect([ 1, 2, 3 ]).to.have.length.below(4); @@ -750,9 +768,20 @@ module.exports = function (chai, _) { function assertBelow (n, msg) { if (msg) flag(this, 'message', msg); - var obj = flag(this, 'object'); - if (flag(this, 'doLength')) { + var obj = flag(this, 'object') + , doLength = flag(this, 'doLength'); + + if (doLength) { new Assertion(obj, msg).to.have.property('length'); + } else { + new Assertion(obj, msg).is.a('number'); + } + + if (_.type(n) !== 'number') { + throw new Error('the argument to below must be a number'); + } + + if (doLength) { var len = obj.length; this.assert( len < n @@ -777,14 +806,15 @@ module.exports = function (chai, _) { /** * ### .most(value) * - * Asserts that the target is less than or equal to `value`. + * Asserts that the number target is less than or equal to `value`. * * expect(5).to.be.at.most(5); * * Can also be used in conjunction with `length` to * assert a maximum length. The benefit being a * more informative error message than if the length - * was supplied directly. + * was supplied directly. In this case the target must + * have a length property. * * expect('foo').to.have.length.of.at.most(4); * expect([ 1, 2, 3 ]).to.have.length.of.at.most(3); @@ -799,9 +829,20 @@ module.exports = function (chai, _) { function assertMost (n, msg) { if (msg) flag(this, 'message', msg); - var obj = flag(this, 'object'); - if (flag(this, 'doLength')) { + var obj = flag(this, 'object') + , doLength = flag(this, 'doLength'); + + if (doLength) { new Assertion(obj, msg).to.have.property('length'); + } else { + new Assertion(obj, msg).is.a('number'); + } + + if (_.type(n) !== 'number') { + throw new Error('the argument to most must be a number'); + } + + if (doLength) { var len = obj.length; this.assert( len <= n @@ -825,14 +866,15 @@ module.exports = function (chai, _) { /** * ### .within(start, finish) * - * Asserts that the target is within a range. + * Asserts that the number target is within a range of numbers. * * expect(7).to.be.within(5,10); * * Can also be used in conjunction with `length` to * assert a length range. The benefit being a * more informative error message than if the length - * was supplied directly. + * was supplied directly. In this case the target must + * have a length property. * * expect('foo').to.have.length.within(2,4); * expect([ 1, 2, 3 ]).to.have.length.within(2,4); @@ -848,9 +890,20 @@ module.exports = function (chai, _) { Assertion.addMethod('within', function (start, finish, msg) { if (msg) flag(this, 'message', msg); var obj = flag(this, 'object') - , range = start + '..' + finish; - if (flag(this, 'doLength')) { + , range = start + '..' + finish + , doLength = flag(this, 'doLength'); + + if (doLength) { new Assertion(obj, msg).to.have.property('length'); + } else { + new Assertion(obj, msg).is.a('number'); + } + + if (_.type(start) !== 'number' || _.type(finish) !== 'number') { + throw new Error('the arguments to within must be numbers'); + } + + if (doLength) { var len = obj.length; this.assert( len >= start && len <= finish @@ -1028,28 +1081,59 @@ module.exports = function (chai, _) { /** - * ### .ownProperty(name) + * ### .ownProperty(name, [value]) * - * Asserts that the target has an own property `name`. + * Asserts that the target has an own property `name` and, optionally, if it has + * (or not, if using `.not`) the desired `value`. * * expect('test').to.have.ownProperty('length'); + * expect('test').to.haveOwnProperty('length'); + * expect('test').to.not.have.ownProperty('foo'); + * expect('test').to.not.haveOwnProperty('foo'); + * expect({ length: 12 }).to.have.ownProperty('length', 12); + * expect({ length: 1337 }).to.not.have.ownProperty('length', 20); + * expect({ length: 12 }).to.haveOwnProperty('length', 12); + * expect({ length: 1337 }).to.not.haveOwnProperty('length', 20); * * @name ownProperty * @alias haveOwnProperty * @param {String} name + * @param {Mixed} value (optional) * @param {String} message _optional_ * @namespace BDD * @api public */ - function assertOwnProperty (name, msg) { + function assertOwnProperty (name, value, msg) { if (msg) flag(this, 'message', msg); var obj = flag(this, 'object'); - this.assert( - Object.prototype.hasOwnProperty.call(obj, name) - , 'expected #{this} to have own property ' + _.inspect(name) - , 'expected #{this} to not have own property ' + _.inspect(name) - ); + var negate = flag(this, 'negate'); + var objHasProperty = Object.prototype.hasOwnProperty.call(obj, name) + var actualValue = obj[name]; + + if (negate && value !== undefined) { + if (actualValue === undefined) { + throw new Error(_.inspect(obj) + ' does not have own property ' + _.inspect(name)); + } + } else { + this.assert( + objHasProperty + , 'expected #{this} to have own property ' + _.inspect(name) + , 'expected #{this} to not have own property ' + _.inspect(name) + ); + } + + if (value !== undefined) { + this.assert( + actualValue === value + , 'expected #{this} to have own property ' + _.inspect(name) + ' of #{exp}, but got #{act}' + , 'expected #{this} to not have own property ' + _.inspect(name) + ' of #{act}' + , value + , actualValue + ); + } + + flag(this, 'object', actualValue); } Assertion.addMethod('ownProperty', assertOwnProperty); @@ -1774,6 +1858,7 @@ module.exports = function (chai, _) { , failNegateMsg , subset , obj + , true ); }); diff --git a/lib/chai/interface/assert.js b/lib/chai/interface/assert.js index 3d1f69295..fa9a3b9d0 100644 --- a/lib/chai/interface/assert.js +++ b/lib/chai/interface/assert.js @@ -431,7 +431,7 @@ module.exports = function (chai, util) { * ### .isNaN * Asserts that value is NaN * - * assert.isNaN('foo', 'foo is NaN'); + * assert.isNaN(NaN, 'NaN is NaN'); * * @name isNaN * @param {Mixed} value @@ -1301,6 +1301,82 @@ module.exports = function (chai, util) { new Assertion(obj, msg).to.not.have.deep.nested.property(prop, val); } + /** + * ### .ownProperty(object, property, [message]) + * + * Asserts that `object` has an own property named by `property`. + * + * assert.ownProperty({ tea: { green: 'matcha' }}, 'tea'); + * + * @name ownProperty + * @param {Object} object + * @param {String} property + * @param {String} message + * @api public + */ + + assert.ownProperty = function (obj, prop, msg) { + new Assertion(obj, msg).to.have.ownProperty(prop); + }; + + /** + * ### .notOwnProperty(object, property, [message]) + * + * Asserts that `object` does not have an own property named by `property`. + * + * assert.notOwnProperty({ tea: { green: 'matcha' }}, 'coffee'); + * + * @name notOwnProperty + * @param {Object} object + * @param {String} property + * @param {String} message + * @api public + */ + + assert.notOwnProperty = function (obj, prop, msg) { + new Assertion(obj, msg).to.not.have.ownProperty(prop); + }; + + /** + * ### .ownPropertyVal(object, property, value, [message]) + * + * Asserts that `object` has an own property named by `property` and a value + * equal to the provided `value`. + * + * assert.ownPropertyVal({ coffee: 'is good'}, 'coffee', 'is good'); + * + * @name ownPropertyVal + * @param {Object} object + * @param {String} property + * @param {Mixed} value + * @param {String} message + * @api public + */ + + assert.ownPropertyVal = function (obj, prop, value, msg) { + new Assertion(obj, msg).to.have.ownProperty(prop, value); + }; + + /** + * ### .notOwnPropertyVal(object, property, value, [message]) + * + * Asserts that `object` has an own property named by `property`, but with a value + * different from that given by `value`. + * + * assert.notOwnPropertyVal({ tea: 'is better'}, 'tea', 'is worse'); + * + * @name notOwnPropertyVal + * @param {Object} object + * @param {String} property + * @param {Mixed} value + * @param {String} message + * @api public + */ + + assert.notOwnPropertyVal = function (obj, prop, value, msg) { + new Assertion(obj, msg).to.not.have.ownProperty(prop, value); + }; + /** * ### .lengthOf(object, length, [message]) * diff --git a/lib/chai/utils/index.js b/lib/chai/utils/index.js index 57ea5ed05..932547d26 100644 --- a/lib/chai/utils/index.js +++ b/lib/chai/utils/index.js @@ -158,3 +158,9 @@ exports.checkError = require('check-error'); */ exports.proxify = require('./proxify'); + +/*! + * isNaN method + */ + +exports.isNaN = require('./isNaN'); diff --git a/lib/chai/utils/inspect.js b/lib/chai/utils/inspect.js index 1a7967732..ef3590cc9 100644 --- a/lib/chai/utils/inspect.js +++ b/lib/chai/utils/inspect.js @@ -4,6 +4,7 @@ var getName = require('./getName'); var getProperties = require('./getProperties'); var getEnumerableProperties = require('./getEnumerableProperties'); +var config = require('../config'); module.exports = inspect; @@ -124,7 +125,15 @@ function formatValue(ctx, value, recurseTimes) { } } - var base = '', array = false, braces = ['{', '}']; + var base = '' + , array = false + , typedArray = false + , braces = ['{', '}']; + + if (isTypedArray(value)) { + typedArray = true; + braces = ['[', ']']; + } // Make Array say that they are Array if (isArray(value)) { @@ -171,6 +180,8 @@ function formatValue(ctx, value, recurseTimes) { var output; if (array) { output = formatArray(ctx, value, recurseTimes, visibleKeys, keys); + } else if (typedArray) { + return formatTypedArray(value); } else { output = keys.map(function(key) { return formatProperty(ctx, value, recurseTimes, visibleKeys, key, array); @@ -228,6 +239,7 @@ function formatArray(ctx, value, recurseTimes, visibleKeys, keys) { output.push(''); } } + keys.forEach(function(key) { if (!key.match(/^\d+$/)) { output.push(formatProperty(ctx, value, recurseTimes, visibleKeys, @@ -237,18 +249,40 @@ function formatArray(ctx, value, recurseTimes, visibleKeys, keys) { return output; } +function formatTypedArray(value) { + var str = '[ '; + + for (var i = 0; i < value.length; ++i) { + if (str.length >= config.truncateThreshold - 7) { + str += '...'; + break; + } + str += value[i] + ', '; + } + str += ' ]'; + + // Removing trailing `, ` if the array was not truncated + if (str.indexOf(', ]') !== -1) { + str = str.replace(', ]', ' ]'); + } + + return str; +} function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) { - var name, str; - if (value.__lookupGetter__) { - if (value.__lookupGetter__(key)) { - if (value.__lookupSetter__(key)) { + var name; + var propDescriptor = Object.getOwnPropertyDescriptor(value, key); + var str; + + if (propDescriptor) { + if (propDescriptor.get) { + if (propDescriptor.set) { str = ctx.stylize('[Getter/Setter]', 'special'); } else { str = ctx.stylize('[Getter]', 'special'); } } else { - if (value.__lookupSetter__(key)) { + if (propDescriptor.set) { str = ctx.stylize('[Setter]', 'special'); } } @@ -318,6 +352,12 @@ function reduceToSingleString(output, base, braces) { return braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1]; } +function isTypedArray(ar) { + // Unfortunately there's no way to check if an object is a TypedArray + // We have to check if it's one of these types + return (typeof ar === 'object' && /\w+Array]$/.test(objectToString(ar))); +} + function isArray(ar) { return Array.isArray(ar) || (typeof ar === 'object' && objectToString(ar) === '[object Array]'); diff --git a/lib/chai/utils/isNaN.js b/lib/chai/utils/isNaN.js new file mode 100644 index 000000000..113256d7f --- /dev/null +++ b/lib/chai/utils/isNaN.js @@ -0,0 +1,26 @@ +/*! + * Chai - isNaN utility + * Copyright(c) 2012-2015 Sakthipriyan Vairamani + * MIT Licensed + */ + +/** + * ### isNaN(value) + * + * Checks if the given value is NaN or not. + * + * utils.isNaN(NaN); // true + * + * @param {Value} The value which has to be checked if it is NaN + * @name isNaN + * @api private + */ + +function isNaN(value) { + // Refer http://www.ecma-international.org/ecma-262/6.0/#sec-isnan-number + // section's NOTE. + return value !== value; +} + +// If ECMAScript 6's Number.isNaN is present, prefer that. +module.exports = Number.isNaN || isNaN; diff --git a/lib/chai/utils/overwriteMethod.js b/lib/chai/utils/overwriteMethod.js index 20657a57c..4c61c181e 100644 --- a/lib/chai/utils/overwriteMethod.js +++ b/lib/chai/utils/overwriteMethod.js @@ -42,7 +42,9 @@ var flag = require('./flag'); module.exports = function (ctx, name, method) { var _method = ctx[name] - , _super = function () { return this; }; + , _super = function () { + throw new Error(name + ' is not a function'); + }; if (_method && 'function' === typeof _method) _super = _method; diff --git a/test/assert.js b/test/assert.js index 8140b7a07..449a1789c 100644 --- a/test/assert.js +++ b/test/assert.js @@ -233,6 +233,9 @@ describe('assert', function () { assert.deepEqual({tea: 'chai'}, {tea: 'chai'}); assert.deepStrictEqual({tea: 'chai'}, {tea: 'chai'}); // Alias of deepEqual + assert.deepEqual([NaN], [NaN]); + assert.deepEqual({tea: NaN}, {tea: NaN}); + err(function () { assert.deepEqual({tea: 'chai'}, {tea: 'black'}); }, "expected { tea: \'chai\' } to deeply equal { tea: \'black\' }"); @@ -329,7 +332,19 @@ describe('assert', function () { }); it('isNaN', function() { - assert.isNaN('hello'); + assert.isNaN(NaN); + + err(function (){ + assert.isNaN(Infinity); + }, "expected Infinity to be NaN"); + + err(function (){ + assert.isNaN(undefined); + }, "expected undefined to be NaN"); + + err(function (){ + assert.isNaN({}); + }, "expected {} to be NaN"); err(function (){ assert.isNaN(4); @@ -338,10 +353,13 @@ describe('assert', function () { it('isNotNaN', function() { assert.isNotNaN(4); + assert.isNotNaN(Infinity); + assert.isNotNaN(undefined); + assert.isNotNaN({}); err(function (){ - assert.isNotNaN('hello'); - }, "expected 'hello' not to be NaN"); + assert.isNotNaN(NaN); + }, "expected NaN not to be NaN"); }); it('exists', function() { @@ -1141,6 +1159,53 @@ describe('assert', function () { }, "blah: expected { a: { b: { c: 1 } } } to not have a deep nested property 'a.b' of { c: 1 }"); }); + it('ownProperty', function() { + var coffeeObj = { coffee: 'is good' }; + + // This has length = 17 + var teaObj = 'but tea is better'; + + assert.ownProperty(coffeeObj, 'coffee'); + assert.ownProperty(teaObj, 'length'); + + assert.ownPropertyVal(coffeeObj, 'coffee', 'is good'); + assert.ownPropertyVal(teaObj, 'length', 17); + + assert.notOwnProperty(coffeeObj, 'length'); + assert.notOwnProperty(teaObj, 'calories'); + + assert.notOwnPropertyVal(coffeeObj, 'coffee', 'is bad'); + assert.notOwnPropertyVal(teaObj, 'length', 1); + + err(function () { + assert.ownProperty(coffeeObj, 'calories'); + }, "expected { coffee: 'is good' } to have own property 'calories'"); + + err(function () { + assert.notOwnProperty(coffeeObj, 'coffee'); + }, "expected { coffee: 'is good' } to not have own property 'coffee'"); + + err(function () { + assert.ownPropertyVal(teaObj, 'length', 1); + }, "expected 'but tea is better' to have own property 'length' of 1, but got 17"); + + err(function () { + assert.notOwnPropertyVal(teaObj, 'length', 17); + }, "expected 'but tea is better' to not have own property 'length' of 17"); + + err(function () { + assert.ownPropertyVal(teaObj, 'calories', 17); + }, "expected 'but tea is better' to have own property 'calories'"); + + err(function () { + assert.ownPropertyVal(teaObj, 'calories', 17); + }, "expected 'but tea is better' to have own property 'calories'"); + + err(function () { + assert.notOwnPropertyVal(coffeeObj, 'sugar', 1337); + }, "{ coffee: 'is good' } does not have own property 'sugar'"); + }); + it('throws / throw / Throw', function() { ['throws', 'throw', 'Throw'].forEach(function (throws) { assert[throws](function() { throw new Error('foo'); }); @@ -1611,6 +1676,14 @@ describe('assert', function () { err(function() { assert.isAbove(1, 1); }, 'expected 1 to be above 1'); + + err(function() { + assert.isAbove(null, 1); + }, 'expected null to be a number'); + + err(function() { + assert.isAbove(1, null); + }, 'the argument to above must be a number'); }); it('atLeast', function() { @@ -1633,6 +1706,13 @@ describe('assert', function () { assert.isBelow(1, 1); }, 'expected 1 to be below 1'); + err(function() { + assert.isBelow(null, 1); + }, 'expected null to be a number'); + + err(function() { + assert.isBelow(1, null); + }, 'the argument to below must be a number'); }); it('atMost', function() { @@ -1872,4 +1952,33 @@ describe('assert', function () { }, 'expected undefined to not be frozen'); }); }); + + it('showDiff true with actual and expected args', function() { + try { + new chai.Assertion().assert( + 'one' === 'two' + , 'expected #{this} to equal #{exp}' + , 'expected #{this} to not equal #{act}' + , 'one' + , 'two' + ); + } catch(e) { + assert.isTrue(e.showDiff); + } + }); + + it('showDiff false without expected and actual', function() { + try { + new chai.Assertion().assert( + 'one' === 'two' + , 'expected #{this} to equal #{exp}' + , 'expected #{this} to not equal #{act}' + , 'one' + , 'two' + , false + ); + } catch(e) { + assert.isFalse(e.showDiff); + } + }); }); diff --git a/test/expect.js b/test/expect.js index 42b50e632..14e6ec2f7 100644 --- a/test/expect.js +++ b/test/expect.js @@ -217,6 +217,34 @@ describe('expect', function () { err(function () { expect([ 1, 2, 3 ]).to.have.length.within(5,7, 'blah'); }, "blah: expected [ 1, 2, 3 ] to have a length within 5..7"); + + err(function () { + expect(null).to.be.within(0, 1, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.be.within(null, 1, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + expect(1).to.be.within(0, null, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + expect(null).to.not.be.within(0, 1, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.not.be.within(null, 1, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + expect(1).to.not.be.within(0, null, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + expect(1).to.have.length.within(5,7, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('above(n)', function(){ @@ -242,6 +270,26 @@ describe('expect', function () { err(function () { expect([ 1, 2, 3 ]).to.have.length.above(4, 'blah'); }, "blah: expected [ 1, 2, 3 ] to have a length above 4 but got 3"); + + err(function () { + expect(null).to.be.above(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.be.above(null, 'blah'); + }, "the argument to above must be a number"); + + err(function () { + expect(null).to.not.be.above(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.not.be.above(null, 'blah'); + }, "the argument to above must be a number"); + + err(function () { + expect(1).to.have.length.above(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('least(n)', function(){ @@ -270,6 +318,26 @@ describe('expect', function () { err(function () { expect([ 1, 2, 3, 4 ]).to.not.have.length.of.at.least(4, 'blah'); }, "blah: expected [ 1, 2, 3, 4 ] to have a length below 4"); + + err(function () { + expect(null).to.be.at.least(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.be.at.least(null, 'blah'); + }, "the argument to least must be a number"); + + err(function () { + expect(null).to.not.be.at.least(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.not.be.at.least(null, 'blah'); + }, "the argument to least must be a number"); + + err(function () { + expect(1).to.have.length.at.least(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('below(n)', function(){ @@ -295,6 +363,26 @@ describe('expect', function () { err(function () { expect([ 1, 2, 3 ]).to.have.length.below(2, 'blah'); }, "blah: expected [ 1, 2, 3 ] to have a length below 2 but got 3"); + + err(function () { + expect(null).to.be.below(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.be.below(null, 'blah'); + }, "the argument to below must be a number"); + + err(function () { + expect(null).to.not.be.below(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.not.be.below(null, 'blah'); + }, "the argument to below must be a number"); + + err(function () { + expect(1).to.have.length.below(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('most(n)', function(){ @@ -324,6 +412,26 @@ describe('expect', function () { err(function () { expect([ 1, 2 ]).to.not.have.length.of.at.most(2, 'blah'); }, "blah: expected [ 1, 2 ] to have a length above 2"); + + err(function () { + expect(null).to.be.at.most(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.be.at.most(null, 'blah'); + }, "the argument to most must be a number"); + + err(function () { + expect(null).to.not.be.at.most(0, 'blah'); + }, "blah: expected null to be a number"); + + err(function () { + expect(1).to.not.be.at.most(null, 'blah'); + }, "the argument to most must be a number"); + + err(function () { + expect(1).to.have.length.of.at.most(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('match(regexp)', function(){ @@ -471,15 +579,63 @@ describe('expect', function () { err(function(){ expect({foo: 'bar'}).to.be.empty; }, "expected { foo: \'bar\' } to be empty"); + + err(function(){ + expect(null).to.be.empty; + }, "expected null to exist"); + + err(function(){ + expect(undefined).to.be.empty; + }, "expected undefined to exist"); + + err(function(){ + expect().to.be.empty; + }, "expected undefined to exist"); + + err(function(){ + expect(null).to.not.be.empty; + }, "expected null to exist"); + + err(function(){ + expect(undefined).to.not.be.empty; + }, "expected undefined to exist"); + + err(function(){ + expect().to.not.be.empty; + }, "expected undefined to exist"); + }); it('NaN', function() { expect(NaN).to.be.NaN; - expect('foo').to.be.NaN; - expect({}).to.be.NaN; + + expect(undefined).not.to.be.NaN; + expect(Infinity).not.to.be.NaN; + expect('foo').not.to.be.NaN; + expect({}).not.to.be.NaN; expect(4).not.to.be.NaN; expect([]).not.to.be.NaN; + err(function(){ + expect(NaN).not.to.be.NaN; + }, "expected NaN not to be NaN"); + + err(function(){ + expect(undefined).to.be.NaN; + }, "expected undefined to be NaN"); + + err(function(){ + expect(Infinity).to.be.NaN; + }, "expected Infinity to be NaN"); + + err(function(){ + expect('foo').to.be.NaN; + }, "expected 'foo' to be NaN"); + + err(function(){ + expect({}).to.be.NaN; + }, "expected {} to be NaN"); + err(function(){ expect(4).to.be.NaN; }, "expected 4 to be NaN"); @@ -487,10 +643,6 @@ describe('expect', function () { err(function(){ expect([]).to.be.NaN; }, "expected [] to be NaN"); - - err(function(){ - expect('foo').not.to.be.NaN; - }, "expected 'foo' not to be NaN"); }); it('finite', function() { @@ -688,15 +840,85 @@ describe('expect', function () { it('ownProperty(name)', function(){ expect('test').to.have.ownProperty('length'); expect('test').to.haveOwnProperty('length'); + expect('test').to.not.have.ownProperty('iDontExist'); + expect('test').to.not.haveOwnProperty('iDontExist'); + expect({ length: 12 }).to.have.ownProperty('length'); + expect({ length: 12 }).to.haveOwnProperty('length'); + expect({ length: 12 }).to.not.have.ownProperty('iDontExist'); + expect({ length: 12 }).to.not.haveOwnProperty('iDontExist'); + + // Chaining property's value + expect('test').to.have.ownProperty('length').that.is.a('number'); + expect('test').to.haveOwnProperty('length').that.is.a('number'); + + err(function(){ + expect({ length: 12 }).to.have.ownProperty('iDontExist'); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function(){ + expect({ length: 12 }).to.not.have.ownProperty('length'); + }, "expected { length: 12 } to not have own property 'length'"); + + err(function(){ + expect({ length: 12 }).to.haveOwnProperty('iDontExist'); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function(){ + expect({ length: 12 }).to.not.haveOwnProperty('length'); + }, "expected { length: 12 } to not have own property 'length'"); + }); + + it('ownProperty(name, value)', function(){ + expect('test').to.have.ownProperty('length', 4); + expect('test').to.haveOwnProperty('length', 4); + expect('test').to.not.have.ownProperty('length', 1337); + expect('test').to.not.haveOwnProperty('length', 1337); + + expect({ length: 12 }).to.have.ownProperty('length', 12); + expect({ length: 12 }).to.haveOwnProperty('length', 12); + expect({ length: 12 }).to.not.have.ownProperty('length', 15); + expect({ length: 12 }).to.not.haveOwnProperty('length', 15); + + // Chaining property's value + expect('test').to.have.ownProperty('length', 4).that.is.a('number'); + expect('test').to.haveOwnProperty('length', 4).that.is.a('number'); var objNoProto = Object.create(null); objNoProto.a = 'a'; expect(objNoProto).to.have.ownProperty('a'); err(function(){ - expect({ length: 12 }).to.not.have.ownProperty('length', 'blah'); - }, "blah: expected { length: 12 } to not have own property 'length'"); + expect({ length: 12 }).to.have.ownProperty('iDontExist', 12); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function() { + expect({ length: 12 }).to.not.have.ownProperty('length', 12); + }, "expected { length: 12 } to not have own property 'length' of 12"); + + err(function() { + expect({ length: 12 }).to.have.ownProperty('length', 15); + }, "expected { length: 12 } to have own property 'length' of 15, but got 12"); + + err(function() { + expect({ length: 12 }).to.not.have.ownProperty('iDontExist', 15); + }, "{ length: 12 } does not have own property 'iDontExist'"); + + err(function(){ + expect({ length: 12 }).to.haveOwnProperty('iDontExist', 12); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function() { + expect({ length: 12 }).to.not.haveOwnProperty('length', 12); + }, "expected { length: 12 } to not have own property 'length' of 12"); + + err(function() { + expect({ length: 12 }).to.haveOwnProperty('length', 15); + }, "expected { length: 12 } to have own property 'length' of 15, but got 12"); + + err(function() { + expect({ length: 12 }).to.not.haveOwnProperty('iDontExist', 15); + }, "{ length: 12 } does not have own property 'iDontExist'"); }); it('ownPropertyDescriptor(name)', function(){ diff --git a/test/should.js b/test/should.js index c32a9e3ac..4995264f9 100644 --- a/test/should.js +++ b/test/should.js @@ -154,12 +154,38 @@ describe('should', function() { }); it('NaN', function(){ - 'foo'.should.be.NaN; + NaN.should.be.NaN; + + Infinity.should.not.be.NaN; + 'foo'.should.not.be.NaN; + ({}).should.not.be.NaN; + should.not.equal(undefined, NaN); (4).should.not.be.NaN; + err(function(){ + NaN.should.not.be.NaN; + }, "expected NaN not to be NaN"); + + err(function(){ + Infinity.should.be.NaN; + }, "expected Infinity to be NaN"); + + err(function(){ + 'foo'.should.be.NaN; + }, "expected 'foo' to be NaN"); + + err(function(){ + ({}).should.be.NaN; + }, "expected {} to be NaN"); + + err(function(){ + should.equal(undefined, NaN); + }, "expected undefined to equal NaN"); + err(function(){ (4).should.be.NaN; - }, "expected 4 to be NaN") + }, "expected 4 to be NaN"); + }); it('undefined', function(){ @@ -235,6 +261,34 @@ describe('should', function() { err(function(){ ({ foo: 1 }).should.have.length.within(50,100, 'blah'); }, "blah: expected { foo: 1 } to have a property 'length'"); + + err(function () { + ('string').should.be.within(0, 1, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.be.within(null, 1, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + (1).should.be.within(0, null, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + ('string').should.not.be.within(0, 1, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.not.be.within(null, 1, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + (1).should.not.be.within(0, null, 'blah'); + }, "the arguments to within must be numbers"); + + err(function () { + (1).should.have.length.within(5,7, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('above(n)', function(){ @@ -254,6 +308,26 @@ describe('should', function() { err(function(){ ({foo: 1}).should.have.length.above(3, 'blah'); }, "blah: expected { foo: 1 } to have a property 'length'"); + + err(function () { + ('string').should.be.above(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.be.above(null, 'blah'); + }, "the argument to above must be a number"); + + err(function () { + ('string').should.not.be.above(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.not.be.above(null, 'blah'); + }, "the argument to above must be a number"); + + err(function () { + (1).should.have.length.above(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('least(n)', function(){ @@ -271,6 +345,22 @@ describe('should', function() { err(function(){ ({foo: 1}).should.have.length.of.at.least(3, 'blah'); }, "blah: expected { foo: 1 } to have a property 'length'"); + + err(function () { + ('string').should.be.at.least(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.be.at.least(null, 'blah'); + }, "the argument to least must be a number"); + + err(function () { + ('string').should.not.be.at.least(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.not.be.at.least(null, 'blah'); + }, "the argument to least must be a number"); }); it('below(n)', function(){ @@ -290,6 +380,26 @@ describe('should', function() { err(function(){ ({foo: 1}).should.have.length.below(3, 'blah'); }, "blah: expected { foo: 1 } to have a property 'length'"); + + err(function () { + ('string').should.be.below(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.be.below(null, 'blah'); + }, "the argument to below must be a number"); + + err(function () { + ('string').should.not.be.below(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.not.be.below(null, 'blah'); + }, "the argument to below must be a number"); + + err(function () { + (1).should.have.length.below(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('most(n)', function(){ @@ -307,6 +417,26 @@ describe('should', function() { err(function(){ ({foo: 1}).should.have.length.of.at.most(3, 'blah'); }, "blah: expected { foo: 1 } to have a property 'length'"); + + err(function () { + ('string').should.be.at.most(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.be.at.most(null, 'blah'); + }, "the argument to most must be a number"); + + err(function () { + ('string').should.not.be.at.most(0, 'blah'); + }, "blah: expected 'string' to be a number"); + + err(function () { + (1).should.not.be.at.most(null, 'blah'); + }, "the argument to most must be a number"); + + err(function () { + (1).should.have.length.of.at.most(0, 'blah'); + }, "blah: expected 1 to have a property 'length'"); }); it('match(regexp)', function(){ @@ -557,16 +687,86 @@ describe('should', function() { it('ownProperty(name)', function(){ 'test'.should.have.ownProperty('length'); 'test'.should.haveOwnProperty('length'); + 'test'.should.not.have.ownProperty('iDontExist'); + 'test'.should.not.haveOwnProperty('iDontExist'); + ({ length: 12 }).should.have.ownProperty('length'); + ({ length: 12 }).should.haveOwnProperty('length'); + ({ length: 12 }).should.not.have.ownProperty('iDontExist'); + ({ length: 12 }).should.not.haveOwnProperty('iDontExist'); ({ 1: 1 }).should.have.ownProperty(1); var objNoHasOwnProperty = {hasOwnProperty: null}; objNoHasOwnProperty.a = 'a'; objNoHasOwnProperty.should.have.ownProperty('a'); + // Chaining property's value + 'test'.should.have.ownProperty('length').that.is.a('number'); + 'test'.should.haveOwnProperty('length').that.is.a('number'); + + err(function(){ + ({ length: 12 }).should.have.ownProperty('iDontExist'); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function(){ + ({ length: 12 }).should.not.have.ownProperty('length'); + }, "expected { length: 12 } to not have own property 'length'"); + + err(function(){ + ({ length: 12 }).should.haveOwnProperty('iDontExist'); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function(){ + ({ length: 12 }).should.not.haveOwnProperty('length'); + }, "expected { length: 12 } to not have own property 'length'"); + }); + + it('ownProperty(name, value)', function(){ + 'test'.should.have.ownProperty('length', 4); + 'test'.should.haveOwnProperty('length', 4); + 'test'.should.not.have.ownProperty('length', 1337); + 'test'.should.not.haveOwnProperty('length', 1337); + + ({ length: 12 }).should.have.ownProperty('length', 12); + ({ length: 12 }).should.haveOwnProperty('length', 12); + ({ length: 12 }).should.not.have.ownProperty('length', 15); + ({ length: 12 }).should.not.haveOwnProperty('length', 15); + + // Chaining property's value + 'test'.should.have.ownProperty('length', 4).that.is.a('number'); + 'test'.should.haveOwnProperty('length', 4).that.is.a('number'); + err(function(){ - ({ length: 12 }).should.not.have.ownProperty('length', 'blah'); - }, "blah: expected { length: 12 } to not have own property 'length'"); + ({ length: 12 }).should.have.ownProperty('iDontExist', 12); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function() { + ({ length: 12 }).should.not.have.ownProperty('length', 12); + }, "expected { length: 12 } to not have own property 'length' of 12"); + + err(function() { + ({ length: 12 }).should.have.ownProperty('length', 15); + }, "expected { length: 12 } to have own property 'length' of 15, but got 12"); + + err(function() { + ({ length: 12 }).should.not.have.ownProperty('iDontExist', 15); + }, "{ length: 12 } does not have own property 'iDontExist'"); + + err(function(){ + ({ length: 12 }).should.haveOwnProperty('iDontExist', 12); + }, "expected { length: 12 } to have own property 'iDontExist'"); + + err(function() { + ({ length: 12 }).should.not.haveOwnProperty('length', 12); + }, "expected { length: 12 } to not have own property 'length' of 12"); + + err(function() { + ({ length: 12 }).should.haveOwnProperty('length', 15); + }, "expected { length: 12 } to have own property 'length' of 15, but got 12"); + + err(function() { + ({ length: 12 }).should.not.haveOwnProperty('iDontExist', 15); + }, "{ length: 12 } does not have own property 'iDontExist'"); }); it('ownPropertyDescriptor(name)', function(){ diff --git a/test/utilities.js b/test/utilities.js index 77db79b01..7201eef6f 100644 --- a/test/utilities.js +++ b/test/utilities.js @@ -302,13 +302,30 @@ describe('utilities', function () { expect(_super).to.be.a('function'); return function () { _.flag(this, 'doesnt', true); - _super.apply(this, arguments); } }); }); var dne = expect('something').to.doesnotexist(); expect(dne.__flags).to.have.property('doesnt'); + + chai.use(function (_chai, _) { + expect(_chai.Assertion).to.not.respondTo('doesnotexistfail'); + _chai.Assertion.overwriteMethod('doesnotexistfail', function (_super) { + expect(_super).to.be.a('function'); + return function () { + _.flag(this, 'doesnt', true); + _super.apply(this, arguments); + } + }); + }); + + var dneFail = expect('something'); + var dneError; + try { dneFail.doesnotexistfail(); } + catch (e) { dneError = e; } + expect(dneFail.__flags).to.have.property('doesnt'); + expect(dneError.message).to.eql('doesnotexistfail is not a function'); }); it('overwriteMethod returning result', function () { @@ -655,6 +672,65 @@ describe('utilities', function () { }); }); + it('inspect every kind of available TypedArray', function () { + chai.use(function (_chai, _) { + var arr = [1, 2, 3] + , exp = '[ 1, 2, 3 ]' + , isNode = true; + + if (typeof window !== 'undefined') { + isNode = false; + } + + // Checks if engine supports common TypedArrays + if ((!isNode && 'Int8Array' in window) || + isNode && typeof 'Int8Array' !== undefined) { + // Typed array inspections should work as array inspections do + expect(_.inspect(new Int8Array(arr))).to.equal(exp); + expect(_.inspect(new Uint8Array(arr))).to.equal(exp); + expect(_.inspect(new Int16Array(arr))).to.equal(exp); + expect(_.inspect(new Uint16Array(arr))).to.equal(exp); + expect(_.inspect(new Int32Array(arr))).to.equal(exp); + expect(_.inspect(new Uint32Array(arr))).to.equal(exp); + expect(_.inspect(new Float32Array(arr))).to.equal(exp); + } + + // These ones may not be available alongside the others above + if ((!isNode && 'Uint8ClampedArray' in window) || + isNode && typeof 'Uint8ClampedArray' !== undefined) { + expect(_.inspect(new Uint8ClampedArray(arr))).to.equal(exp); + } + + if ((!isNode && 'Float64Array' in window) || + isNode && typeof 'Float64Array' !== undefined) { + expect(_.inspect(new Float64Array(arr))).to.equal(exp); + } + }); + }); + + it('truncate long TypedArray', function () { + chai.use(function (_chai, _) { + + var arr = [] + , exp = '[ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, ... ]' + , isNode = true; + + // Filling arr with lots of elements + for (var i = 1; i <= 1000; i++) { + arr.push(i); + } + + if (typeof window !== 'undefined') { + isNode = false; + } + + if ((!isNode && 'Int8Array' in window) || + isNode && typeof 'Int8Array' !== undefined) { + expect(_.inspect(new Int8Array(arr))).to.equal(exp); + } + }); + }); + it('addChainableMethod', function () { chai.use(function (_chai, _) { _chai.Assertion.addChainableMethod('x',