Promised-based Asynchronous Test Runner (patr) is a very simple, easy-to-use test runner that support asynchronous JavaScript testing with promises. Patr is based on the premise that testing should be as simple as creating an object with methods to perform tests. Objects can be nested to create subgroups of tests. Patr relies on the system's "assert" module for making assertions. An example test:
var assert = require("assert");
require("patr/runner").run({testMath: function(){
assert.equal(3, Math.min(3, 5));
});
The suggested pattern for writing test files is to define the tests on the exports object and running the test runner if the module is the main module. This allows for direct execution of test files and easy inclusion of tests into other test groups. For example, we could define "my-math-test.js":
var assert = require("assert");
exports.testMath = function(){
assert.equal(3, Math.min(3, 5));
};
if(require.main == module)
require("patr/runner").run(exports);
Now we can directly execute "my-math-test.js" or we could include it in another test file:
exports.mathTests = require("my-math-test");
exports.otherTests = require("other-tests");
if(require.main == module)
require("patr/runner").run(exports);
With these aggregate test module, each test file's tests are included in a nested object that will be tested as a subgroup of tests.
Promises make asynchronous testing very simple. You simply return a promise from your test to indicate when a test is completed. When using promise-based coding this is super simple. For example, to test the contents of file loaded asynchronously using promised-io's fs module:
var fs = require("promised-io/fs");
exports.testFile = function(){
return fs.readFile("testfile").then(function(contents){
assert.equal(contents.toString(), "expected contents");
});
};
...
Patr includes an "assert" module (patr/assert) that is upgraded for promise-based asynchronous code blocks. In particular, the "throws" method can be used to enforce that a code block will eventually throw (or reject) even if it happens asynchronously. For example:
var assert = require("patr/assert");
exports.testFile = function(){
return assert.throws(function(){
// asserts that this must throw/reject eventually
return fs.readFile("non-existent file");
});
};
You can include additional testing options by setting flags on the test objects (that have the test functions). The test object can have the following properties:
- iterations - The number of times to execute the test
These properties are inherited, if they are set by a parent object, than the children objects will inherit the behavior unlesss it is overriden.
These properties can also be set from the command line. This is done by including -name value arguments when starting the test module. For example:
nodules test.js -iterations 1000
Patr is part of the Persevere project, and therefore is licensed under the AFL or BSD license. The Persevere project is administered under the Dojo foundation, and all contributions require a Dojo CLA.