As you know, to have automated visual regression tests (in addition to unit and integration tests) is important to ensure the users a good experience.
This assures that nothing is accidentally broken and confirms that user flows and the appearance of your application are functional when the existing code is modified or a dependency is updated.
Recently, I wrote an article on visual regression testing with TypeScript, Jest, Jest Image Snapshot and Puppeteer.
If you use Puppeteer, you know that you can test only under Chromium.
The implementation with Firefox is experimental, and Webkit is not supported at all.
Today, I will offer you an alternative to Puppeteer: Playwright
Playwright is pretty similar to Puppeteer. All the features present on Puppeteer are not yet present on Playwright. But the advantage with this tool is that you can run your tests on Chromium, Firefox, Webkit, Google Chrome & Microsoft Edge, and in addition with a single command.
In this article, I will explain how to set up and to run tests with Playwright, in a TypeScript project, using Jest - as the test runner - and Jest-Image-Snapshot (Jest matcher) - to take screenshots of current web pages and to compare generated screenshots with a screenshot baseline to find regressions in the user interface.
If you read my previous article, it will be pretty similar.
I will use the BPMN Visualization project (version 0.13.0) as an example. (This example has been simplified so it shows more clearly the configuration and features explained in this article.) The goal of this project is to load BPMN content, and render it.
Automated visual tests will simplify our life with each refactoring, addition of a new component, update of the positioning algorithm of the different BPMN elements, or update of the MxGraph rendering library.
Also, I will configure the tests to run on Chromium, Firefox and Webkit.
As first step, we need to install the required packages as
Jest is a fully featured testing framework, developed by Facebook. It needs very little configuration and works basically out of the box.
npm install -D jest @types/jest
- Playwright: A Node.js library to automate Chromium, Firefox,Webkit, Google Chrome & Microsoft Edge with a single API. It is possible to perform most of the actions that are done manually on a browser and take screenshots.
npm install -D playwright jest-playwright-preset
- Jest-Image-Snapshot + its type definition: A Jest matcher to perform image comparisons
npm i -D jest-image-snapshot @types/jest-image-snapshot
Let’s configure the previous libraries.
I won't go into detail here on all the different ways to configure Jest.
If you already use Jest for your unit/e2e tests, this is not new for you. If you would like more explanations about Jest, there are many great articles available.
In this example, we have 4 Jest configurations: unit tests, integration tests, e2e tests and performance tests. We added the visual tests in the e2e test suite. Here, I will just explain how we configure Jest for the e2e tests.
First, create the Jest configuration file (jest.config.js) at ./test/e2e directory:
This configuration sets the root directory to the root project directory, runs .ts files with ts-jest module and looks for .spec.ts and test.ts files under any subdirectory of ./test/e2e directory.
Specify the preset in the Jest configuration (./test/e2e/jest.config.js), as specified in the official documentation of Playwright:
Create a new file (./test/e2e/jest-playwright.config.js) for the Playwright configuration to run the server & launch the browser once for all tests:
With this configuration, we start a server on the port 10002 with a timeout of 30s, and start 3 browsers.
This is the part that might be new, but with a little configuration, we will be ready soon.
By default, Jest doesn’t know anything about Jest-Image-Snapshot and its assertion toMatchImageSnapshot. So we’ll need to extend Jest. For that, create a new file (./test/e2e/jest.image.ts), like the following:
To avoid extending Jest in each test file or import the previous file globally in all test files, we need to configure Jest to run it immediately after the test framework has been installed in the environment with setupFilesAfterEnv (Jest property) in ./test/e2e/jest.config.js.
To simply the test execution, add the following script in the package.json file:
Now, you can run your e2e tests with the following command:
npm run test:e2e
Note: cross-env is useful if you run the tests on different OS.
You can find the different properties to customize Jest-Image-Snapshot in its README on Github.
If everything is configured correctly, we are now ready to create our first visual regression test (./test/e2e/bpmn.rendering.test.ts) by combining Playwright and Jest and Jest-Image-Snapshot!
After the test runs, a new directory will be created -
__image_snapshots__ - with an image for each toMatchImageSnapshot call. The name of the snapshot is computed by default with testPath, currentTestName, counter and defaultIdentifier.
Example of generated snapshot:
Note: Make sure that the snapshot files are committed in your source control so that they are shared with other developers and CI environments.
One issue with one-to-one pixel matching is that there is a good chance that the test will be in error on a machine other than on which it was developed, because every environment has slightly different ways of rendering the same application.
For example, suppose that we want to run the tests on the CI environment every time we create a pull request to the master branch in GitHub.
Without any modifications to the code, the test is passed locally; but on the CI environment, it fails with a message like this:
Error: Expected image to match or be a close match to snapshot but was 0.0005804554357724534% different from snapshot (2.7861860917077763 differing pixels).
And a new image file for the diff is stored in the
__image_snapshots__/__diff_output__ directory with the name <snapshot_name>-diff.png.
You can modify the previous jest-image-snapshot configuration (./test/e2e/bpmn.rendering.test.ts), and update the value of failureThreshold (default value: 0) & failureThresholdType (default value: pixel). These properties are used to calculate the threshold of tolerated differences (before the test fails).
Warning: If you increase the failure threshold too much, when there is too much difference between local and CI environments, it may be impossible to detect visual regressions.
If you have 10 or more tests, it can become complicated to find which screenshot corresponds to which test / feature in the directory __image_snapshots__.
Modify the customSnapshotsDir property to have a different value according to the tests.
Sometimes the expected result/snapshot is the same even after different actions. To avoid having a lot of identical snapshots in the Github repository, it’s better to reuse a snapshot.
For that, it’s necessary to override the default customSnapshotIdentifier & customDiffDir properties.
customSnapshotIdentifier: the custom name to give this snapshot. This prevents the name of the snapshots from being computed with testPath, currentTestName, counter and defaultIdentifier.
customDiffDir: the custom absolute path of a directory to keep this diff in. As we use the same snapshot in different tests, to know which diff file corresponds to which test, we need to set a different value according to the tests.
With so many operating systems, web browsers, and screen resolutions, Visual Testing can be a powerful tool to assure that an application works well in all possible environments. It is definitely worth trying it as a complement to other sets of tests.
Now, you have everything you need to start your first visual regression test in TypeScript with Jest & Playwright.
Thank you for reading and I hope I helped or inspired you :)
Github repository of BPMN Visualization: https://github.com/process-analytics/bpmn-visualization-js
Github repository of Jest-image-snapshot: https://github.com/americanexpress/jest-image-snapshot
Playwright documentation: https://playwrightdev/
Main image made with: https://www.canva.com