Workflow Automation with Grunt
I recently came across Grunt.js while working on Erdős at SDSLabs. It is a task-based command line tool similar to Brunch, Maven, Gradle, and I love it! It’s primarily used to automate repetitive tasks in the development workflow and any (lazy) developer would fall in love with it. It takes care of tasks like compilation, minification, concatenation, versioning, cache-busting, linting, unit testing, etc. It isn’t limited to JS-specific tasks either, and it’s fairly easy to leverage the power of Node by using child processes. Also, it is well supported by an extremely active plugin development community. There may be other Javascript task runner solutions, but I don’t know of any at the moment that are worth taking a look at. In this post, I’ll go through the setup and a few Grunt tasks that I find really useful.
The basic setup
As I’ve mentioned earlier, Grunt uses Node.js and is installed via NPM (Node Package Manager). Once you’ve got those ready, install the Grunt CLI globally.
The second command should output your current command line module version. Next, create package.json
, which keeps track of the dependencies we’re using in our project so we don’t have to push the node modules when collaborating with other developers.
This is what the project directory looks like which I’ll be referring to throughout this post.
my-test-app/
|--assets/
|--|--dist/
|--|--style.css
|--|--main.js
|--|--jquery.js
|--index.tmpl
|--package.json
|--gruntfile.js
package.json
Run npm install
, and NPM will go fetch these for us and place them in a node_modules
folder (which should ideally be under gitignore).
Tasks
Create gruntfile.js
which defines the workflow and tasks for Grunt to execute. Inside gruntfile.js
, all configuration is done by passing an object literal to grunt.initConfig()
.
Matchdep
We are using matchdep to load all the grunt-modules in a single line instead of loading them sequentially. That is to say that instead of using:
We are using the matchdep module to retrieve the dependency list from package.json, filter it for grunt-*
, and load all those modules in grunt using:
CSS Minification
This is used to minify all the CSS files for use in the production environment.
This task can now be executed by calling grunt cssmin:production
on the command line.
Uglify JS
This takes care of javascript compression and minification, thus reducing file size which is important for websites to load fast.
This task can now be executed by running grunt uglify:production
from the command line. It will concatenate and uglify jquery.js
and main.js
in one file.
Environment Configuration
This is a grunt task to automate environment configuration for other tasks. It can be used with the grunt-preprocess plugin to build index.html
on-the-fly.
Preprocess
As mentioned earlier, Grunt can preprocess files based off environment configuration.
This task can now be executed by calling grunt preprocess
on the command line. Our layout.tmpl
can have logical blocks. This is useful in including analytics only on the production build, throwing in some ascii art, changing static asset paths based on environment etc.
Cache-busting
grunt-hashres is an extremely useful plugin that hashes js and css files and renames the <script>
and <link>
declarations that refer to them in my html/php/etc files.
This task can now be executed by running grunt hashres:production
from the command line. It would change <link rel="stylesheet" href="assets/dist/style.css">
to <link rel="stylesheet" href="assets/dist/style.130fdfaa.css">
and <script src="assets/dist/main.js">
to <script src="assets/dist/main.9c4cc83e.js">
in index.html
. It would also rename the corresponding CSS & JS files in the assets/dist
directory. This is much better than using a timestamp because the hash only changes when a file has been updated.
Build
Public grunt tasks defined at the bottom of gruntfile.js
can be called directly from the command line. The default
task is executed by running grunt
. Environment variables should ideally be picked up from the project configuration file. Here is the complete gruntfile for convenience. If everything went right, you should see the following output when you run grunt
.
This was just a basic introduction and walkthrough to using Grunt. Actual usage would definitely be more robust and complex. Grunt comes with pretty much everything you’ll need to use it on a large project and can be extended as much as you want.