The following is a guest post by Garris Shipon. We’ve touched on the four types of CSS testing here before. Regression testing is the hardest. It’s the type where you’re trying to test if a change you made to CSS resulted in any unexpected visual problems. This is made more difficult with responsive designs. Garris built a tool for doing this as he embarked upon a new responsive design for a large scale site. Here, he’ll walk you through the whole thing.
This article was originally published in December 2014. It has been re-written, updated, and republished now in April 2016 because BackstopJS, the main tool presented here, as been updated.
A use-case for visual regression testing
Do a search for “CSS regression testing” and a common theme becomes clear: breaking CSS is easy, testing it is hard.
This was the case at the onset of a responsive CSS refactoring project I scoped for a large online retailer. Like many other web companies at the time, we were in the process of adding responsive behavior to a massive e-commerce web app, which was originally designed for 1024px desktop screens.
I realized this would be a regression-prone job. Retrofitting multiple breakpoint behaviors meant we would likely have a lot of hard-to-find display bugs. I needed a way for our engineers to automate bug discovery before slamming our QA team with hundreds of ticky-tacky little layout issues.
Where BackstopJS fits in
The solution I wanted had to play nice with web developers. That is, easy to install locally, use familiar web dev paradigms, and give a reasonable amount of confidence that a selector change made for mobile isn’t going to result in a hard-to-find bug in a desktop layout.
At the time, there wasn’t anything out of-the-box that quite fit the bill. This was the reason for creating BackstopJS.
BackstopJS is a visual regression testing app which wraps CasperJS, PhantomJS and ResembleJS in an easy-to-configure test matrix across multiple app-states (URLs), DOM elements and screen sizes.
The following is a 15 minute walk-through of an installation and initial configuration of BackstopJS.
A visual regression-test tutorial
This instructional will be based on a simple demo project (download ZIP here). It is taken directly from the Bootstrap example page.
Expand the simple demo
Unzip the project download. We will install the testing framework right into this example project:

Here is what you’ll see if you open up myCoolProject/index.html
in a web browser… Remember, it’s responsive. So, make sure to resize the browser window down to see the most narrow layout!

Install BackstopJS with NPM
Now go to your project root (`/myCoolProject/`) and run:
$ cd ~/path-to-myProjects/myCoolProject
$ npm install backstopjs
Your directory should now look like this:

Install complete! Now let’s get to some basic testing…
Generating a BackstopJS configuration template
The basic configuration process is straightforward from here. To help with things, BackstopJS can generate a config file that you can modify for your project. From the `myCoolProject/node_modules/backstopjs/` directory run:
$ cd ~/path-to-myProjects/myCoolProject/node_modules/backstopjs/
$ npm run genConfig
This will add files to your project root: folders for BackstopJS screenshots, `backstop_data`, and generating a boilerplate configuration file `backstop.json`.

The configuration file is where you’ll specify your testing rules. Let’s look at that file.
{
"viewports": [
{
"name": "phone",
"width": 320,
"height": 480
}, {
"name": "tablet_v",
"width": 568,
"height": 1024
}, {
"name": "tablet_h",
"width": 1024,
"height": 768
}
],
"scenarios": [
{
"label": "My Homepage",
"url": "http://getbootstrap.com",
"hideSelectors": [],
"removeSelectors": [
"#carbonads-container"
],
"selectors": [
"header",
"main",
"body .bs-docs-featurette:nth-of-type(1)",
"body .bs-docs-featurette:nth-of-type(2)",
"footer",
"body"
],
"readyEvent": null,
"delay": 500,
"onReadyScript": null,
"onBeforeScript": null
}
],
"paths": {
"bitmaps_reference": "../../backstop_data/bitmaps_reference",
"bitmaps_test": "../../backstop_data/bitmaps_test",
"compare_data": "../../backstop_data/bitmaps_test/compare.json",
"casper_scripts": "../../backstop_data/casper_scripts"
},
"engine": "phantomjs",
"report": ["browser", "CLI"],
"cliExitOnFail": false,
"debug": false,
"port": 3001
}
In this configuration you can see three viewports
objects. One for phone, tablet vertical, and tablet horizontal, each with name and dimensions properties. You can add as many viewports
objects as you need. BackstopJS requires at least one.
Then we have scenarios
which include the URLs and element selectors that BackstopJS will test. It’s useful to think of every scenario object as a test for a specific static page or global app state. Add as many scenarios
as you need. BackstopJS requires at least one.
Inside each scenario is a list of selectors. Selectors accept standard CSS notation. For each selector you specify, BackstopJS will take a screenshot and test that area of your layout. Your selector area could be as small as a button
or as big as the body
of your page — check the documentation for more on using this feature with dynamic web apps.
Modifying the configuration template
For our demo, make the following change and replace the scenarios
node in `myCoolProject/backstop.json`.
"scenarios": [
{
"label": "My Local Test",
"url": "../../index.html",
"hideSelectors": [],
"removeSelectors": [
],
"selectors": [
"nav",
".jumbotron",
"body .col-md-4:nth-of-type(1)",
"body .col-md-4:nth-of-type(2)",
"body .col-md-4:nth-of-type(3)",
"footer"
],
"readyEvent": null,
"delay": 0,
"onReadyScript": null,
"onBeforeScript": null
}
],
Generating reference screenshots
From the `myCoolProject/node_modules/backstopjs/` directory run…
$ npm run reference
This task will create (or update an existing) screen captures representing all specified selectors at every breakpoint. When the process is complete, take a look inside `/myCoolProject/backstop_data/bitmaps_reference/`:

So far so good. We have our reference set. Now let’s run a test!
Running our first test
We are about to run our first test. But keep in mind, we haven’t changed anything in our CSS yet, so our tests should pass!
From the `myCoolProject/node_modules/backstopjs/` directory run:
$ npm run test
This task will create a new, timestamped-directory of test images inside `/myCoolProject/backstop_data/bitmaps_test/<timestamp>`.
Once the test images are generated, BackstopJS will open your web browser and display a report comparing the most recent test bitmaps against the current reference images. Significant differences (if any) are detected and shown.

In this instance, since we haven’t made any changes to our test page, BackstopJS should show all of our tests as passing. Now, let’s try changing our CSS and see what happens.
Updating our index file and running our second test
Here is what you’ll see if you open up our (unchanged) `myCoolProject/index.html` in a web browser. Notice the margin around the text:

Let’s mess that up! Open up `myCoolProject/index.html` and insert the following code just before the closing </head>
tag:
<style>
.jumbotron {
padding: 0px;
}
</style>
Here’s what the page looks like now:

This is exactly the kind of thing that happens all the time during web development. Some unscoped code gets in and hoses your layout just enough that you might not notice :(
Now, From the `myCoolProject/node_modules/backstopjs/` directory run:
$ npm run test
Our test should run again and errors should be found. Scroll the report down to see a visual diff of the issues we’ve just created…

Our visual diff contains the reference capture, the most recent test capture and the visual diff file.

There you have it: regression found!
This is a very simple example. In real life, designers and engineers may find themselves working on very large and or complex CSS projects. That is when a system like this really improves the quality and consistency of our work. By automating the repetitive visual tasks we can confidently pursue more creative ones.
About workflow
There are many ways to integrate this kind of test into your workflow. You could fire off tests every time you build or you could manually run tests (as you work or just before pushing to your next stage). You could even integrate BackstopJS into your CI pipeline, if that’s your thing. All of these topics are outside the scope of this article. Check the documentation for more info.
Next steps
Since first releasing in 2014, BackstopJS has grown substantially. There are loads of newly added features developed by the community:
- SPA testing support – Use Casper scripts and explicit web app triggers to ensure screenshots are captured at the correct time inside your web app (e.g. after API responses, after CSS animation completion, or wait for any other watchable async process).
- Simulating user interactions – Use Casper scripting inside your scenarios to simulate interactions with your on-screen components.
- CI pipeline integration – BackstopJS CLI features have enabled advanced users to make visual regression testing a part of their continuous integration process.
- Active configuration files – Enables active (node module) logic inside your config files which you can use to change testing behavior based on environment or other conditions, point to different config files to use your BackstopJS instance as a centralized test server for multiple environments, verticals, profiles, projects, or whatever.
More on BackstopJS
- BackstopJS.org
- Find documentation, file bugs, get troubleshooting help, learn about advanced features and contribute on GitHub!
I am continually amazed at how much there is to learn about web development. However, this still feels very exciting.
Amazing stuff, can’t wait to check this out. I really like where the web is going.
Tried it and loved it.
Cant wait for more browser engine support.
I ran in to one problem installing.
–> sudo npm install phantomjs
i had to do install it globally “-g”
–> sudo npm install -g phantomjs
@Kenneth take a look at Gemini (https://github.com/bem/gemini) for broader support
Thanks Bro to Help me
Cool story bro. Too bad about that document.ready() bit, that’s a dealbreaker here. Looking forward to next year’s update!
Same here. I was really excited until I read that last paragraph. Cool tool though. Looking forward to the future improvements.
Yup, I know — it’s actually a huge deal breaker for me on most of my projects and I built the tool! I am planning to add this as the very next update. Something like a listener on the console and/or the window which could be configured with …
BackstopJS would wait to run a test if consoleOrWindowReadyEvent is truthy. So your app would decide when it’s view is sufficiently loaded and emit (e.g. ‘everythingIsReady’) on the console or window node — then the snapshots would run.
Any comments or suggestions here would be great. And of course any contributions here would be stellar. ;)
What do you think about Gemini?
The best tool for css regression at the moment :)
http://sourcejs.com/ is planning to add support for gemini too.
How do you think about https://github.com/winsonwq/viff
I’m confused why so many regression testing frameworks require a user to globally install modules such as Phantom and Casper on their machine. I feel this is a lot to ask of a user, especially if you would like to integrate such a solution in a CI environment. I’ve made the following plugins, loosely based on PhantomCSS with the option to use ResembleJS for image comparison. The CLI uses Phantom and Casper executables rather than global install:
https://www.npmjs.org/package/resemble-cli
https://www.npmjs.org/package/grunt-resemble-cli
Your framework is by far better than these (as they don’t screenshot components, only full pages), but this also allows the option for image comparison using GraphicsMagick (which must be globally installed). If there is one module I would want users to globally install it would be GM rather than Phantom or Casper as I found image comparison in Resemble is extremely slow and in GM it is almost instant.
Also, why not publish this to NPM as well and save the users the step of having to cd into a Bower repo and npm install?
Finally! I was dreaming about something like that, but was to lasy to make own research. Now, when it was given almost on the plate I will totally make an use of it. TY!
Visual Regression Testing is so, so, so useful on large sites, and especially on large teams. I wrote up my experiences with implementing it at http://sonniesedge.co.uk/2014-10-17/visual-regression-testing.html
Charlie, I really like the process detail and thoughts in your article as well! I would also be interested in how this might be combined with grabbing components for a style guide, as well as a workflow utilizing uncss. I’m new to Grunt/Gulp, so hope I’m not stepping in too deep. Chris, you rock as usual!
Visual Regression Testing, if for testing in a blog like wordpress how. is there a tutorial here on discussing visual testing on wordpress?
thank you
The issue I’ve found with image based testing is that it doesn’t work well with dynamic content. On a recent project made up of multiple single page apps, we used the Galen Framework for a hybrid approach: image comparison for static components and UI measurements, content testing and css rule checks via Selenium. Galen handles both. It does require some configuration, but it also works with Browserstack/SauceLabs to help avoid wrangling vms and to provide testing on real devices.
In terms of complete test coverage, I haven’t found anything comparable.
I am by no means a master as you are. But for “unexpected visual problems” I have done a few tests on changes in the past and tried in various browsers etc. – only to receive emails shortly afterwards after applying the changes to a website, from users telling me they had such and such problem with site’s appearance in EXACTLY the same browser I had already “satisfactorily” tested with. However, the bitmaps reference is of course still a brilliant idea to catch stuff I might not even have “seen” in the first place!
+1
Site under test normally has a dynamic content. It means the content can be changed and regression tests will fail. I solve it by keeping on dev environment a styleguide section. Those are static HTML pages with every single styled component on them. Besides test-ability, it also a good proof that CSS in general is designed decently. Being a sort of a gallery, the styleguide is quite handy when picking a component for reuse. Moreover, technically that makes it close to unit-testing. Regressively or not, that way, in fact, we test every component styles independently and get notified of a problem regardless of context.
Here the walk-through:
http://dsheiko.com/weblog/automated-css-regressive-testing-in-practice/
Is there a version of this tool that does not require NPM and Node?
I have no interest at this point in time to try Node, and no plans in the future, but it seems like a lot of neat tools get built with it.
Gemini is probably the standout right now. It utilises Selenium and PhantomJS, enabling cross-browser regression testing.
https://github.com/gemini-testing/gemini
Now that’s awesome!
Great post and thanks for sharing Garris Shipon.
Cheers!
I tried using this, but when running
‘npm run reference’
I get the following error:
”’
BackstopJS Config loaded at location D:\www\demo\backstop.json
[09:34:41] Using gulpfile D:\www\demo\node_modules\backstopjs\gulpfile.js
[09:34:41] Starting ‘echo’…
Running CasperJS with: [ ‘capture/echoFiles.js’ ]
[09:34:41] Finished ‘echo’ after 16 ms
Echo files failed with code: 1
”’
I’m running this on Windows 10 with NPM and NodeJS installed
I have the same error as you. Did you find some way to solve it?
Please post any issues on github. Thanks! https://github.com/garris/BackstopJS/issues
Way to go Garris!