In this article we’ll learn how to use Grunt, a task runner, to automate vital web development tasks so we can forget about them. Grunt can do repetitive tasks involving concatenation, compilation, minification, listing, unit-testing, live-reloading, etc. It does these tasks with one script and can be shared among every person working on a project. This ensures that everyone working on the project is writing code of a consistent standard. It’s also easy to work with, so let’s get started.


To get Grunt set up, we’ll need Node.js installed. You don’t need to know Node to use Grunt; you just need it installed. With Node installed, run this command in the terminal:

npm install -g grunt-cli // Try "sudo npm install -g grunt-cli" if it fails without

This command tells npm (a package manager that comes with Node) to install the grunt command-line package. This puts the grunt command in out system path. -g stands for “global” and allows this command to be run from any directory.

Setting up a new project

To get Grunt working in our project, we will need two files:

  1. package.json
  2. Gruntfile.js

The package.json file in our projects root directory enables us to track and install all our development dependencies. Add the following text into the package.json file.

  "name" : "YOUR_GRUNT_PROJECT",
  "version" : "0.1.0",
  "author" : "YOUR NAME",
  "devDependencies" : {
    "grunt" :  "~0.4.5"

We will also need to configure Grunt to use the plugins, so create a new file in the projects root directory called Gruntfile.js and add these lines of code:

module.exports = function(grunt){

  // Configuration
    // Read from package.json
    pkg: grunt.file.readJSON('package.json')

  // Tells Grunt what commands to run when we type “grunt” in the terminal
  grunt.registerTask('default', []);

Run the following command to install the dependencies listed in the package.json file and save them in a node_modules folder:

npm install

Now that we have set up Grunt, all Grunt plugins can be installed by following these steps:

  1. Find the plugin and install it from the command line.
  2. Learn and configure it using our Gruntfile.js file.

Concatenate JavaScript

Let’s assume that our project directory looks something like this:

- js
  - plugins
    - modernizr.js
    - jquery.js
    - lightbox.js
  - main.js
- css
  - base.scss
  - layout.scss
- img
  - cat.jpg
  - another_cat.jpg
  - facepalm.gif
  - sprite.png
- index.html
- folders...

We probably have multiple JavaScript files in our project. It’s a good idea to concatenate these files in production to lower the number of HTTP requests required to render the page. We can set up Grunt to do this for us automatically. We can install the concatenation plugin by running this command from our projects root directory:

npm install grunt-contrib-concat --save-dev

This will download the required files in the npm_modules directory and adds "grunt-contrib-concat": "~0.5.0" into the devDependencies section of our package.json file.

For the file structure above, Grunt concatenation can be configured using our Gruntfile.js file like this:

module.exports = function(grunt){

  // Configuration
    // Read from package.json
    pkg: grunt.file.readJSON('package.json'),
    concat: {
      dist: {
        src: [
        'js/plugins/*.js', // Specifies all of the files with the .js extension in the plugins folder
        'js/main.js'  // The main JS file
        dest: 'js/build/main.js', // The final build JS file

    // Tells Grunt to load the plugin we just installed

    // Tells Grunt what commands to run when we type “grunt” in the terminal
    grunt.registerTask('default', ['concat']);

Minify JavaScript

First we install the uglify plugin from the command line:

npm install grunt-contrib-uglify --save-dev

Then we configure it in our Gruntfile.js file:

uglify: {
  build: {
    src: 'js/build/production.js', // Our un-minified production JS file
    dest: 'js/build/production.min.js' // Automagically minified JS file!

Don’t forget that we need to load the task and register it under our default task in our Gruntfile.js file:


grunt.registerTask('default', ['concat', 'uglify']);

Compile Sass

If you use a preprocessor (and you should – it’s helpful), Grunt has a plugin that will automate preprocessing for us. Assuming you already have Sass installed, we can install the Grunt plugin using the following command:

npm install grunt-contrib-sass --save-dev

Then we load and register it:


grunt.registerTask('default', ['concat', 'uglify', 'sass']);

Now we configure it:

sass: {
  dist: {
    options: {
      style: 'compressed'
      files: {
        'css/build/main.css': 'css/*.scss'

Optimize images

Install it:

npm install grunt-contrib-imagemin --save-dev

Load and register it:


grunt.registerTask('default', ['concat', 'uglify', 'imagemin']);

Configure it:

imagemin: {
  dynamic: {
    files: [{
      expand: true,
      cwd: 'img/',
      src: ['**/*.{png,jpg,gif}'],
      dest: 'img/build/'

Watch and LiveReload webpages

With Grunt we can “watch” files using the grunt-contrib-watch plugin. You know the drill, install it, load it, and register it.

Here’s how to configure it:

watch: {
  scripts: {
    files: ['js/*.js'],
    tasks: ['concat', 'uglify'],
    options: {
      spawn: false,

We can take this a step further and enable the LiveReload ability which comes with the watch plugin.

To get LiveReload working, we will need to:

  1. Install the free LiveReload browser extension
  2. Click the LiveReload button in our browser to activate it.
  3. Add a few lines to the watch configuration in the Gruntfile.js file:
watch: {
  options: {
    livereload: true,
    scripts: {
      // Rest of the code


Grunt can help us automate a lot out our projects, helping us focus on the fun stuff. There are many more plugins available here. A few of my favorites that I didn’t mention already are cssmin, jshint, csslint, autoprefixer, htmlmin, svgmin, and copy. Have fun!