Using headless Chrome as an automated screenshot tool

Starting with Chrome 57, we can run Chrome as a headless browser on linux servers. Capturing web page screenshots is a common use case for browser automation, and PhantomJS has always been my go-to solution for this, but let’s explore what this looks like using Node.js to drive headless Chrome:

And the setup required to run this code on a linux server (I used a Vagrant with Ubuntu Trusty Tahr):

Voila!

eff.org captured via headless chrome from a linux server

A couple thoughts after going through this process:

  • Chrome headless is not perfect yet. For example, I can run the same command twice in a row and one screenshot looks perfect, the other is strangely cropped (although it’s possible my code is to blame).
  • Tools like Nightmare or Phantom can still exist. Why? As of right now, the Chrome debugging protocol API and chrome-remote-interface bindings aren’t perfectly optimized for developer use cases. You’ll notice in our example there is some boilerplate required, and there is a lot of async code, which would’ve been painful to write without async/await. Passing around nodeIds for the DOM methods is also not ideal. There is still room for a library with a better API that uses headless Chrome as the underlying engine.

All that said, I’m very optimistic about headless Chrome and believe that it’ll soon be the most reliable engine for browser automation on the server. For those that would like to start testing it as a screenshot tool, you can clone or fork this repo which contains the example code found in this post.