Run mocha tests via concurrent grunt tasks.
npm install grunt-mocha-concurrent --save-dev
Even though there are lots of posts online that don't recommend running tests in parallel, I've almost halved the time it takes for my tests to run. Sure, it can be harder to debug issues, but since this uses grunt-concurrent
to spawn separate processes there won't be any "cross wire" bugs. Besides, if your test fails you can just revert to running your regular serial grunt test task along with a mocha .only
. But when things are green then why not go faster? 👍
This plugin requires grunt-mocha-test
and grunt-concurrent
. grunt-env
is optional.
Check that your Gruntfile.js
looks something similar to this. It's important that Concurrent.init(grunt);
appears after you call initConfig()
.
const Concurrent = require("grunt-mocha-concurrent");
...
grunt.loadNpmTasks('grunt-mocha-test');
grunt.loadNpmTasks('grunt-concurrent');
grunt.loadNpmTasks('grunt-mocha-concurrent');
...
grunt.initConfig({
...
});
Concurrent.init(grunt);
If your tests don't have any central external dependencies like a database, then this is the example for you.
grunt.initConfig({
...
mocha_concurrent: {
your_target: {
specs: [
{
mochaSpec: {
options: {
reporter: 'dot',
timeout: '2000'
},
src: ['tests/unit/**/test*.js']
}
},
{
mochaSpec: {
options: {
reporter: 'dot',
timeout: '2000'
},
src: ['tests/functional/**/test*.js']
}
}
]
},
}
...
});
...
grunt.registerTask('spec:concurrent', ['mocha_concurrent:your_target']);
When you run grunt spec:concurrent
it will run your unit tests and your functional tests in parallel.
mochaSpec
is just forwarded directly togrunt-mocha-test
so look up their documentation if you want to know more about the options.
Say your functional tests read and/or write to the database. You can't run parallel tests in the same process as your tests will step all over each other. You can't run parallel tests in different processes pointing to the same database either! So we get around this by using a different database instance in each task. ✌️
If you use an environment variable to set which database to connect to, then you can use this example. This assumes that your test framework will dynamically create your database/schema for you. I'm using mongodb so this just happens automagically.
grunt.initConfig({
grunt.loadNpmTasks('grunt-env');
...
mocha_concurrent: {
your_target: {
specs: [
{
mochaSpec: {
options: {
reporter: 'dot',
timeout: '2000'
},
src: ['tests/unit/**/test*.js']
}
},
{
mochaSpec: {
options: {
reporter: 'dot',
timeout: '2000'
},
src: ['tests/functional/controllers/**/test*.js']
},
envSpec: {
MONGODB_URI: 'mongodb://localhost:27017/my-test-db-1'
}
},
{
mochaSpec: {
options: {
reporter: 'dot',
timeout: '2000'
},
src: ['tests/functional/models/**/test*.js']
},
envSpec: {
MONGODB_URI: 'mongodb://localhost:27017/my-test-db-2'
}
}
]
},
}
...
});
...
grunt.registerTask('spec:concurrent', ['mocha_concurrent:your_target']);
When you run grunt spec:concurrent
it will run your unit tests and two of your functional test flavours in parallel.
grunt.initConfig({
mocha_concurrent: {
options: {
concurrentLimit: 1, // Number of concurrent tasks to run in parallel. Defaults to number of cpu cores.
envDefault: {}, // Environment variables to pass to all tasks. Defaults to null.
envTaskPrefix: "", // Prefix to use for grunt-env tasks. Defaults to mochaConcurrent-.
mochaTaskPrefix: "", // Prefix to use for grunt-env tasks. Defaults to mochaConcurrent-.
},
...
},
});
- You can get fancy and generate your mocha_concurrent target specs depending on what you need. E.g. I looped over each folder under
tests/functional/controllers
and created a separate spec for each, each with a unique database connection string. - If your application is a HTTP API and you call it in your tests, you can also just increment the port number for each spec using the env variable.
- There is overhead involved in spinning up separate tasks with their own database etc. The more cores you have the more time this will save. Otherwise it may actually be slower than a single task!
- Clean up old databases after the tests to save space. I defined a global mocha post test using
.after()
and included it in the mochaSpec src for each spec. Mongoose gives you a way to delete the database. - If you don't want grunt to stop when it encounters a failed test, consider using
grunt-continue
.