I'm not really planning anything big in the near future, but if I embark on building something that is more than a prototype or a hack I'm going have to unit test. So tonight I'm going to have a quick poke around to see what folks are using for unit testing node application.
I'm not sure what the de facto framework is, but I'm going to follow the pattern set over the years of _something_Unit. So, I'm going to punt at there being a NodeUnit. Google "nodeunit" -> Bang!
There is already a good article on getting started, so I'm not going to add any value here tbh. But I'll carry on and record my concise notes, just in case I need to refer to them myself in the future.
I'm on Ubuntu, though it shouldn't matter which platform you're on. (if truth be known, I've had some issues with code being cross-platform compatible with node)
$ sudo npm install nodeunit -g
I thought I'd knock up something that is at least vaguely security related. I'm not getting into arguments around random passwords vs memorable passwords, nor am I starting on hashing, salting or key stretching; this is simply something that allows me to test drive nodeunit quickly and simply.
The example is a simple, old-skool password policy. Stuff like:
- Minimum length
- Upper case
- Lower case
- Numeric
- Special characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//validator.js | |
//This is not a recommended password policy, this is purely an example that allows me to use nodeunit | |
var minimumLength = 12; | |
var minimumCriteria = 4; //if you only want to impose 2 of the 4 possible criteria, simply change this to 2 | |
var hasNumeric = /\d+/; | |
var hasUpper = /[A-Z]+/; | |
var hasLower = /[a-z]+/; | |
var hasSpecial = /[!\"£$%^&\*()_\+\-={}\[\];'#:@~,\.\/<>\\?\|]/; | |
//use the unary plus to convert from true/false to 1/0 | |
//could use the double bitwise not too ~~ | |
exports.validate = function (value) { | |
if (typeof value !== 'string') { | |
throw new Error('Expected a string'); | |
} | |
if (value.length < minimumLength) { | |
throw new Error('String too short'); | |
} | |
if (((+hasNumeric.test(value)) + | |
(+hasUpper.test(value)) + | |
(+hasLower.test(value)) + | |
(+hasSpecial.test(value))) < minimumCriteria) { | |
return false; | |
} | |
return true; | |
}; |
- If minimum length is not met, throw an error
- If it is not a string, throw an error
- If the criteria is not met, fail the test
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//test_validator.js | |
var validator = require('./validator'); | |
exports['validate'] = function (test) { | |
//these tests need to fail with an error being thrown | |
test.throws(function () { validator.validate(); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(null); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(true); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate([]); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate({}); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(0123); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(0123456); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(01234567); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(012345678); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(0123456789); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate(01234567890); }, 'Expected a string', 'value passed is not a string'); | |
test.throws(function () { validator.validate('a'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('abcd'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('abcdefg'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('abcdefghi'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('A'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('ABCD'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('ABCDEFG'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('ABCDEFGHI'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('ABCDEFGHIJK'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('0'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('0123'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('0123456'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('012345678'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('01234567891'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('!$%#'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('!"£$%^&'); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('!"£$%^&*('); }, 'String too short', 'less than min length'); | |
test.throws(function () { validator.validate('!"£$%^&*()_'); }, 'String too short', 'less than min length'); | |
//these tests need to fail | |
test.ok(!validator.validate('abcdefghijkl'), 'only lower case'); | |
test.ok(!validator.validate('ABCDEFGHIJKL'), 'only upper case'); | |
test.ok(!validator.validate('abcdefGHIJKL'), 'only lower and upper case'); | |
test.ok(!validator.validate('012345678910'), 'only numerics'); | |
test.ok(!validator.validate('abcdef012345'), 'only lower case and numeric'); | |
test.ok(!validator.validate('GHIJKL012345'), 'only upper case and numeric'); | |
test.ok(!validator.validate('abcdeFGHIJ01234'), 'only lower, upper case and numeric'); | |
test.ok(!validator.validate('!"£$%^&*()_+'), 'only specials'); | |
test.ok(!validator.validate('abcdef!"£$%^'), 'only lower case and specials'); | |
test.ok(!validator.validate('ABCDEF!"£$%^'), 'only upper case and specials'); | |
test.ok(!validator.validate('abcdefABCDEF!"£$%^'), 'only lower, upper case and specials'); | |
test.ok(!validator.validate('012345!"£$%^'), 'only numberics and specials'); | |
test.ok(!validator.validate('012345ABCDEF!"£$%^'), 'only upper case, numerics and specials'); | |
test.ok(!validator.validate('abcdef012345!"£$%^'), 'only lower case, numerics and specials'); | |
//these tests need to pass | |
test.ok(validator.validate('Ab0!cdefghij'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('aBcd0!fghijk'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('Abcd0!efghijkl'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('01234!fghIJKL0123'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('!fghIJKL0123'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('fghIJKL0123!'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('KL01234!fghij'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('!"£$%fghijABCDE01234'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0!bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0"bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0£bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0$bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0%bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0^bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0&bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0*bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0(bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0)bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0_bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0+bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0-bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0=bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0{bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0}bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0[bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0]bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0;bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0\'bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0#bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0:bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0@bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0~bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0,bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0.bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0/bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0<bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0>bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0\\bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0?bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
test.ok(validator.validate('A0|bcdeFGHIJ'), 'string, good length, upper, lower, numeric, special'); | |
for(var i = 0; i < 500000; i++) { | |
test.ok(validator.validate(generatePassword()), 'string, good length, upper, lower, numeric, special'); | |
} | |
test.done(); | |
}; | |
//could have used something like this, to produce an alphnumeric: Math.random().toString(35).slice(-12); | |
var namespace = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz!\"£$%^&\*()_\+\-={}\[\];\'#:@~,\.\/<>\\?\|'; | |
function generatePassword() { | |
var password = namespace.charAt(Math.floor(Math.random() * 10)); | |
password += namespace.charAt(10 + Math.floor(Math.random() * 26)); | |
password += namespace.charAt(36 + Math.floor(Math.random() * 26)); | |
password += namespace.charAt(62 + Math.floor(Math.random() * 32)); | |
do { | |
password += namespace.charAt(Math.floor(Math.random() * namespace.length)); | |
} while (password.length < 12); | |
return password; | |
} | |
As an aside, I kinda like the fact that the validator is validating the test utility that generates passwords to be used in the tests, code validating the validation tests...
Anyhoo; running the tests...
$ nodeunit test_validator.js
The code ran first time, surprising since I wasn't using an IDE, it was the old fave TextPad. More surprising though was how quickly it ran; I upped the random generating of passwords to 5,000,000 and it took only 15 seconds to execute. Cool.
So, to summarise my fleeting introduction to nodeunit.
- It's fast
- Easy to install
- Quick to install
- Easy to understand
- Fast
There are mode frameworks out there too, but from experience, you tend to stick with what works until it doesn't work. So, for me, I think nodeunit will be my de facto node...unit.