rabid.audio

Documenting my work at the intersection of technology and music.

Using Grunt with Jekyll

Published: 19 Dec 2014

Note

This is an old post or draft which was migrated from my old blog. It may have broken links, and it definitely has questionable opinions. Consume at your own risk.

As you can see, I finally have my new blog going. I was originally going to write a Jekyll-like CMS with Node and Backbone, but I finally gave in and used the real thing. It’s a lot more flexible than I expected, and over all I’m loving it. There was just one thing I learned in my previous attempt that I really wanted, and that was Grunt.

If you aren’t familiar, grunt is a script runner for javascript, analogous to rake in the Rails world. My first experience with it was trying to fix a broken Gruntfile from a Yeoman generator, and I was quite intimidated. But I’ve picked up enough now to know it is really not so bad. I’m going to build up my Gruntfile for Jekyll step-by-step, and hopefully explain how to use Grunt in the process.

Preparation

You want to start by making a package.json file for your project if you don’t already have one. This will be useful for saving all the npm packages your grunt needs to work for anyone who clones/forks your code at a later date. Simply run

npm init

and follow the prompts.

You then need to install the command line interface globally, so that you can run it from the terminal:

npm install -g grunt-cli

And you need to install an instance of grunt locally:

npm install --save-dev grunt

The --save-dev will save this as a required development package for your project. Other developers can use npm install --dev to install all the dependencies. This is separate from --save so that you can separate the executing dependencies of your project (perhaps connect or express if you are running a web server) from the packages useful for developing the project (your test suite, for example).

Now you need to make a file called Gruntfile.js in the root directory of your project. This is where you define all of the tasks you can run. Here’s the skeleton:

module.exports = function(grunt) {

  grunt.initConfig({

  });

  grunt.registerTask('default', []);
});

Adding Tasks

Jekyll

Common Grunt tasks are shared via npm. All(?) of them have a name that starts with grunt-. For example, here’s one that’s really useful for me: grunt-jekyll. It allows me to run jekyll build from inside a grunt task. Since this is one of the most important things to do in my project, that’s pretty important.

We start by installing the package, the same way we did with grunt:

npm install --save-dev grunt-jekyll

Now there are three things we need to do to get this task configured in our Gruntfile.js. First, we need to load the task with

grunt.loadNpmTasks('grunt-jekyll');

This makes the package available to grunt. Then we need to add the configuration. If we look at the documentation, it tells us all the configuration options for this package. Inside grunt.initConfig, we need to add a jekyll configuration. Each item inside is a specific task.

grunt.initConfig({
  jekyll: {
    //Production Build
    dist: {
      options: {
        dest: '_site/blog',
        config: '_config.yml'
      }
    },
    //Develpemnt Build
    dev: {
      options: {
        dest: '_site/blog',
        config: '_config.yml',
        drafts: true
      }
    }
  }
});

Here I have two jekyll tasks, one for each target. On my production builds, I want to simply do jekyll build, but for development builds, I want to also include drafts so I can see what my current draft looks like. I tell it which configuration file to use. Now you can run both of these tasks:

grunt jekyll:dist
grunt jekyll:dev

If we simply run grunt jekyll, it will run all the targets, one after another. Now, there are a lot more options we can set for this task, but I’m using mostly the default options, so we can simplify this significantly:

jekyll: {
  dist: {},
  dev: { options: { drafts: true } }
}

Here’s our whole file at present:

module.exports = function(grunt) {

  //load tasks
  grunt.loadNpmTasks('grunt-jekyll');

  //configure tasks
  grunt.initConfig({
    jekyll: {
      dist: {},
      dev: { options: { drafts: true } }
    }
  });

  grunt.registerTask('default', []);
});

We will talk about that last line, grunt.registerTask, in just a minute, but to see it’s value, let’s add another task first.

JSHint

Let’s add another step to our build process, and lint our javascript with jshint. I keep my custom javascript files in _src and my third party files from Bower in _vendor, for reasons which will be more apparent shortly. I’m assuming you are familiar with jshint already. To install another task, it’s the same process: install, load, configure:

npm install --save-dev grunt-contrib-jshint
grunt.loadNpmTasks('grunt-contrib-jshint');
grunt.initConfig({
  ...
  jshint:{
    src: ['_src/**/*.js']
  }
});

I have a .jshintrc file already in my project, so it knows what rules to follow. I made one task, src, which takes as options an array of files or directories to lint, in this case any javascript file inside my _src directory (no need to lint bower packages). Since there is only one task, we can run this like

grunt jshint

Nice! Now, while I’m developing and constantly testing some javascript, it would be annoying to have jshint run after every change, but I would like it to run before I push a real production build. But running

grunt jshint && grunt jekyll:dist

is kind of annoying. The real value of a task runner is that you can string together multiple tasks. So let’s make a build and a devbuild task, the first which runs jshint and then jekyll:dist and the second which runs jekyll:dev:

grunt.registerTask('build', ['jshint', 'jekyll:dist']);
grunt.registerTask('devbuild', ['jekyll:dev']);
grunt.registerTask('default', []);

We simply registerTask with the name of the task and an array of subtasks to run in the order we want to run them. Simple and awesome. That default one is special, it’s the task to run if we don’t specify a task. Let’s set it to run build:

grunt.registerTask('default', ['build']);

Now try it with

grunt

and it should run jshint first and then jekyll:dist.

Divshot

Now that we’ve got the hang of this, I’m going to quickly throw in a deploy task. GitHub has free hosting for Jekyll blogs, but I’m currently trying out Divshot, which is kind of similar to Heroku but for static sites. There’s a package already, grunt-divshot, which I install, load, and make some default tasks for. This package has a webserver built in, but I’m going to use connect, so I simply don’t use the divshot task (which runs divshot:server), and instead make a separate task, divshot:push, with two sub-targets, development and production. No need to configure anything because it’s already in my divshot.json config file. This dimonstrates another cool feature of grunt: tasks are infinitely sub- dividable. To deploy to production would be grunt divshot:push:production while to push to both would be runt divshot:push. And naturally, I make two deploy tasks which build first:

grunt.registerTask('devdeploy', ['devbuild', 'divshot:push:development']);
grunt.registerTask('deploy', ['build', 'divshot:push:production']);

Javascript Compiling

Including a whole bunch of separate javascript files can really slow down your page load. Every separate resource (individual js or css file, image, etc.) causes the browser to make a separate HTTP request, opening a separate TCP connection to the server, creating all this overhead to send a few kilobytes. With sass (already built into Jekyll), all our stylesheets get compiled into a single css/main.css file. We can do a similar thing for our javascript, compressing it in the process. Two popular packages are grunt-contrib-concat, which will concatenate files together, and grunt-contrib-uglify, which can combine and compress javascript files to a single minified script. On production builds, I want my output javascript file (which I’m keeping in js/output.js) to be minified for speed. But when developing, I want un-minifed versions of my javascript so I can step through it with the debugger if I need it. So I use concat for development builds and uglify for production. Same song: install, load, and configure.

grunt.initConfig({
  ...
  //js compilation (production version)
  uglify: {
    dist: {
      files: {
        'js/output.js': ['<%= cfg.vendor %>', '_src/**/*.js']
      },
      options: {
        compress: true,
      }
    }
  },

  //js compilation (dev version)
  concat: {
    options: {
      separator: '\n\n\n',
      banner: '/*! <%= pkg.name %> - v<%= pkg.version %> - ' +
        '<%= grunt.template.today("yyyy-mm-dd") %> */',
    },
    dev: {
      src: ['<%= cfg.vendor %>', '_src/**/*.js'],
      dest: 'js/output.js',
    },
  },
});

The <%= %> parts are templates. Grunt will execute what’s inside as javascript and put the result there. We can grab variables from JSON or YAML files as such:

grunt.initConfig({

  //config files
  pkg: grunt.file.readJSON('package.json'),
  cfg: grunt.file.readYAML('_config.yml'),

  ...

In this case, I’m reading from package.json for some variables and _config.yml for others. In my _config.yml, I made a list of the javascript files from my bower packages to include, called vendor, so once I’ve loaded the config file, I can access this with cfg.vendor. Grunt is smart enough to take the string '<%= cfg.vendor %>', replace it with my YAML array, and then flatten the whole files array so that output.js is made of first my included bower javascript files, followed by all javascript files inside the _src directory. Note that while uglify has one task, dist, with options, concat has an options object which applies to all tasks (of which there is only dev). Most grunt tasks allow for global options which can be overwritten by individual tasks.

Advanced Tasks

Up to this point, all these tasks have been pretty simple. You might be wondering why you’d bother with grunt at all and not just make some shell scripts. Well, one benefit is cross-compatibility; any platform with Node.js can use it. Another is the ability to make hierarchies of tasks. To do the same in scripts would require one script for each task or a bunch of function definitions. Still, that’s a pretty reasonable point. The next task is something that would be significantly more work to script but is painless in grunt. But first, let’s add some cool features to our existing Gruntfile.

Autoload Tasks and Time Them

Right now, we have a stack of these loadNpmTasks commands:

grunt.loadNpmTasks('grunt-jekyll');
grunt.loadNpmTasks('grunt-contrib-jshint');
grunt.loadNpmTasks('grunt-divshot');
grunt.loadNpmTasks('grunt-contrib-concat');
grunt.loadNpmTasks('grunt-contrib-uglify');

This could potentially get really long if we keep adding tasks at this rate. Of course, all of these packages are listed in our package.json file, since we are using --save-dev. So there’s a neat package load-grunt-tasks which will run all of these automatically. Just install it and replace all of these loadNpmTasks statements with a single line:

require('load-grunt-tasks')(grunt);

Now, even if we install more grunt tasks, we don’t need to load them. We can simply skip that step and jump straight to the configuration.

Here’s another little one, time-grunt, which will tell us the execution time of every grunt task. Simply install it and add

require('time-grunt')(grunt);

Watch and Serve Locally

Now the real magic happens. One of the most popular grunt tasks is grunt-contrib-watch, which allows you to run tasks when files change. We can also add a local server to see what the site looks like currently. Since we’re using Node, we can use the amazing connect, specifically the grunt task implementation, grunt-contrib-connect. Combining these two together with a feature connect-livereload, we can make a task which starts a webserver and listens for file changes, automatically rebuilding the project and refreshing the page in the browser.

npm install --save-dev grunt-contrib-watch grunt-contrib-connect connect-livereload
grunt.initConfig({
  ...
  watch: {
    options:{
      spawn: false,
      livereload: grunt.option('livereloadport') || LIVERELOAD_PORT
    },
    scripts: {
      files: ['_src/**/*.js', '_vendor/**/*.js'],
      tasks: ['devbuild'],
    },
    content: {
      files: [
        '*.*',
        '_sass/*css',
        'css/*css',
        '_drafts/*',
        '_posts/*',
        '_layouts/*',
        '_includes/*',
        'images/*',
        '_plugins/*',
        '_config.yml'
      ],
      tasks: ['jekyll:dev'],
    },
  },

  //serve it up
  connect: {
    options: {
      port: grunt.option('port') || SERVER_PORT,
      hostname: '0.0.0.0'
    },
    livereload: {
      options: {
        middleware: function (connect) {
          return [
            require('connect-livereload')({port: LIVERELOAD_PORT}),
            mountFolder(connect, '_site')
          ];
        }
      }
    }
  }
});

Let’s break this down. watch has some global options and two file sets to watch, scripts which runs our whole devbuild task including recompiling our javascript, while content just runs jekyll:dev. To get connect to host the livereload script, we need to return require('connect-livereload')({port: LIVERELOAD_PORT}) as the first middleware object. Meanwhile, mountFolder is a simple function at the top of the script to serve up static files:

var mountFolder = function (connect, dir) {
    return connect.static(require('path').resolve(dir));
};

Finally, we tack on grunt-open and we can even automatically open the site in a new tab.

open: {
  server: {
    path: 'http://localhost:<%= connect.options.port %>/blog'
  }
}

Now that we have all of these partial tasks, let’s make the default task open the server and watch for changes.

grunt.registerTask('serve', ['devbuild', 'connect:livereload', 'open:server', 'watch']);
grunt.registerTask('default', ['serve']);

Custom Tasks

Last but not least, let’s add a custom task for making new posts. There’s a sublime package that will let you do this, but my installation of sublime is buggy and plugins mess up. To make a custom task, simply call registerTask with the task name, a description, and a callback to execute. Any subtasks will be included as arguments to the function, e.g. ‘new:post’ will call the new task with the first argument "post".

grunt.registerTask('new', 'Start a new post or draft', function(type) {
  ...
});

I made a function to ask for the post title and copy a file _templates/post.md to the correct folder (_draft or _post) with the correct name and current date. Then using grunt-open it opens this file in your preferred text editor (from package.json or the $EDITOR environment variable).

Here’s the link to my complete Gruntfile.js if you want to see it all. Looking at a complete file like this can be pretty intimidating, but hopefully by building it up a little bit at a time you can build one for your next project.