Test screens in acceptance/integration tests for visual changes over time.
ember install ember-visual-test
Use ember-visual-test in your acceptance tests like this:
import { test } from 'qunit';
import moduleForAcceptance from '../../tests/helpers/module-for-acceptance';
import { capture } from 'ember-visual-test/test-support/helpers';
moduleForAcceptance('Acceptance | visual test');
test('visiting /', async function(assert) {
await visit('/');
assert.equal(currentURL(), '/');
await capture(assert, 'test-file-name');
});
Whenever capture
is called in the test, the node server will make a screenshot with
simple-headless-chrome,
and save it in the /visual-test-output/baseline
folder. Please commit this folder into source control!
Now, whenever the test is run, a new snapshot is made and put in the /visual-test-output/tmp
folder
(do NOT put that into source control!). It then compares the two images with
pixelmatch and asserts accordingly.
If a mismatch is found, it will save an image with the diff of the two versions in the /visual-test-output/diff
folder, to help you identify the issue.
Note that this means that if a screen changes consciously, you'll need to either manually
delete that image from the /visual-test-output/baseline
folder,
or run ember visual-test:reset
to reset ALL images.
For example, imagine you had an image like this the first time you ran the capture:
Now, after some changes, you run the tests again, and this time, it captures this image:
This would result in an error, and generate the following diff image:
Sadly, the font rendering is slightly different on different platforms (Windows/Linux/MaxOS). This can lead to comparison errors when comparing a baseline image generated on one platform, with one on another platform. This is especially problematic on CI (e.g. Travis or CircleCI), if your local machine doesn't run on Linux.
To prevent wrong errors to pop up there, this addon will by default namespace the images with the OS, and thus generate a different set of baseline (and comparison) images for windows/linux/mac. Of course, this also means that if you generate baseline images on your PC, and then run tests on Linux, it will not actually compare the images, and will always succeed. However, this is still preferable to tests always failing. You can also put multiple different baseline images into Git, to let the test compare to the fitting one.
For information on how to turn on the by-OS comparison, see the settings section below.
The capture function takes three parameters: capture(assert, identifier, options)
assert
: The assert function. This assumes you are using qunit.identifier
: A unique string to identify this capture. This will be the file name of the generated images, and has to be unique across your whole application.options
: An optional object with options. The following options are allowed:selector
: An optional selector to screenshot. If not specified, the whole page will be captured.fullPage
: If a full page screenshot should be made, or just the browsers viewport. Defaults totrue
.delayMs
: Delay (in milliseconds) before taking the screenshot. Useful when you need to wait for CSS transitions, etc. Defaults to100
.
This works in both acceptance tests as well as in integration tests.
There are a few configuration options for you ember-cli-build.js
, with the default values:
let app = new EmberAddon(defaults, {
visualTest: {
imageMatchAllowedFailures: 0, // How many failed pixels result in an error
imageMatchThreshold: 0.3, // This is a config option for pixelmatch
imageLogging: false, // If generated images should be logged to the console
debugLogging: false, // If console messages from headless chrome should be printed in the console
imgurClientId: null, // If set to a client ID of imgur, images will be uploaded there as well, to debug images e.g. on CI
groupByOs: true, // If one set of images should be created/compared by OS
noSandbox: false // This may need to be set to true depending on your environment e.g. in CI
}
});
Also note that for the capture helper to work, you'll need to include the babel polyfill, at least in the test environment:
'ember-cli-babel': {
includePolyfill: true
}
If you are using ember-cli-mirage for acceptance testing
you'll need to add a passthrough for /visual-test/make-screenshot
to your project's mirage/config.js
.
this.passthrough('/visual-test/make-screenshot');
There are also two CLI commands to use:
ember visual-test:clean
: Clean the diff/tmp foldersember visual-test:reset
: Manually clean all folders & create new baseline images
When running in capture mode (in node), the capture
helper will automatically add a class
visual-test-capture-mode
to the body. Styles are injected when running tests that use that
class to hide the qunit-specific chrome & make the test view full screen.
If you want to "ignore" some element in the capture (e.g. because it changes in every rendering),
you can add the attribute data-test-visual-ignore
to it, which will give it an opacity of 0
when run in capture mode.
This is how this addon actually works:
- When the app is started, we boot an instance of headless Chrome.
- A middleware is registered which captures ajax POSTs to
/visual-test/make-screenshot
- This middleware takes an URL as parameter, and instructs Chrome to visit that URL and make a screenshot of it.
- Now, on the client/test side of things, the
capture()
helper makes an ajax POST to the middleware, giving it the URL of the current test (via testId). Additionally, it appends a custom&capture=true
parameter. So basically, we instruct the headless chrome to view the same test/page we are currently at. - The
capture()
helper checks if thecapture=true
param is set, and if so, it will pause the test and emit an event so that the headless chrome knows it is ready. - It also adds a class to the body which is used to remove the test-specific wrappers etc. to get a nice full screen capture.
Please note that since this relies quite heavily on node, it requires node >= 8 to take advantage of newer JavaScript features.