The other day at work a colleague shared the basics on how to start doing Accessibility testing using JAWS and WAVE. It was quite motivating so I started looking into how we can have some of these tests automated and found out that there is actually a way to run automated accessibility tests that could provide a good amount of coverage in a really fast manner.
This could potentially benefit the rest of the team to allow them to have an extra tool that could quickly provide an accessibility assessment, at least for all possible tests covered by the Axe-Core Library.
Anyways, I took some time today to draft this quick POC on how to run some accessibility tests with Cypress and Axe-Core. I did have to spend some time on getting some better-formatted results so this is actually something useful.
To get started we'll need the following installed on the machine where you plan to run this POC:
- JavaScript IDE: Visual Studio Code, Atom, etc.
- NodeJS LTS Version Installed
- Cypress latest version.
- Chrome or any other browser supported by Cypress.
If you want to just run this solution instead of creating it from scratch, you can just follow these simple steps:
- Clone the Repository:
git clone https://github.com/tzero86/CypressAxe
- Access cloned folder and Install the dependencies:
npm install
- In the
Cypress.json
file, set the URL you want to test. - Run the tests by doing:
npm run test
- Alternatively you can run the test using Cypress Runner:
npx cypress open
Happy testing!
Let's start by setting up NodeJS. Depending on your OS you'll grab the corresponding installer from here. In my case I'm running Manjaro Linux, so I had to run this: sudo pacman -S nodejs
and sudo pacman -S npm
. But it any case the installation is well documented on NodeJS' website so we won't spend any more time on this.
Now we need to start a new project, for that you'll run the following command in a folder of your choosing: npm init -y
to initialize the project. In my case I ran these commands already from VSCode integrated terminal.
Next is setting up Cypress, and that just requires running the following in the terminal: npm install cypress
Once that is done, we run the following to start Cypress: npx cypress open
.
With this we have Cypress ready and we can just close it out for now. Next we need to install Axe-Core plugin for Cypress. In the same terminal we run the following command: npm install cypress-axe
and then npm install axe-core
After that our package.json file should look something like this:
We are almost ready to start creating our automated accessibility testing POC, we just need one last detail. We need to update Cypress' support files so it loads the cypress-axe commands for us to use and also so Cypress can log the results of the testing done by Axe. We need to import Cypress-axe into cypress/support/index.js
file in the import commands section, you just need to add import 'cypress-axe'
. This will cause that all Cypress-axe commands get accessible from the cy
object exposed by Cypress.
That's it, now let's move to the actual testing with these tools we just set.
To get things started we need to create a new file: ada_poc.spec.js
, inside this file we'll add our sample test and configure the rest so we can have a working prototype of how we can integrate accessibility testing into an E2E automation solution. Even if we don't want to create a whole E2E automation suite, we can still have a very fast test that will run on whatever page we need it to scan and will return a list of all the accessibility issues detected.
NOTE: It is important to understand that this testing POC WONT cover all possible accessibility issues and WONT replace manual ADA testing. As per Axe's library, it should cover around 55% of the Accessibility assessments without any false positives.
Let's add one generic sample test to our recently created file:
// FILE: cypress/integration/ADA_POC/ada_poc.spec.js
describe('Accessibility testing POC', ()=>{
it('This page should be accessible.', ()=>{
cy.visit('www.google.com');
cy.injectAxe();
cy.checkA11y();
})
})
And if we run npx cypress open
and click on our test we see the following:
It seems to be working, it found 5 issues! It is important to mention that Cypress-axe will run, by default, on the entire URL we pass to Cypress. But it can be configured to scan or even exclude desired elements on the page.
However, the results are not that useful right now. We know that Axe seems to have detected some accessibility issues on the target page, but we definitely need better results so we can assess the impact of the issues, determine if we need to assign those to QA for further testing with JAWS/WAVE to confirm the impact, have those reported as bugs and also to be able to provide some details to the DEV team to review the problem more easily.
In order to get better results out of the automated testing we configured using Cypress and Axe-core, we need to do some tweaking. We need to add some code in order for Cypress to be able to log the results from Axe in a more user-friendly-shareable format. Open the following file: cypress/plugins/index.js
, notice this is not the same file we edited before. Once opened we'll be adding the following piece of code. This code will create a custom command and two helper functions that will pretty-format the test results so the results can be easily reviewed in both console and the Cypress runner.
The custom command connects all the parts and will let us even further simplify our tests, since we can focus just on the logic required for the test itself leaving all helper functions under cypress configuration files. Then by calling this single command called testAccessibility
, the code will automatically take care of instructing Cypress to navigate to the target page, will inject Axe into the page and then run the Axe-Core tests for WCAG A and AA tests automatically.
// FILE: cypress/plugins/index.js
module.exports = (on, config) => {
on('task', {
log(message) {
console.log(message)
return null
},
table(message) {
console.table(message)
return null
}
})
}
Now we need to add some more code into the following file: cypress/support/commands.js
, this is how the file should look like:
const indicators = {
critical: '🟥',
serious: '🟧',
moderate: '🟨',
minor: '🟩',
}
function logViolations(violations) {
terminalLog(violations)
violations.forEach(violation => {
const nodes = Cypress.$(violation.nodes.map(node => node.target).join(','))
let log = {
name: `[${indicators[violation.impact]} ${violation.impact.toUpperCase()}]`,
consoleProps: () => violation,
$el: nodes,
message: `[${violation.help}](${violation.helpUrl})`
}
Cypress.log(log)
violation.nodes.forEach(({ target }) => {
Cypress.log({
name: '-🩸FIXME',
consoleProps: ()=> violation,
$el: Cypress.$(target.join(',')),
message: target
})
})
});
}
const terminalLog = (violations) => {
cy.task(
'log',
`\n${'TEST RESULTS'}
\n${violations.length} accessibility violation${
violations.length === 1 ? '' : 's'
} ${violations.length === 1 ? 'was' : 'were'} detected\n`
)
cy.log('log', violations)
const violationData = violations.map(
({ id, impact, description, nodes, help, helpUrl}) => ({
QUANTITY: nodes.length,
IMPACT: `${indicators[impact]} ${impact.toUpperCase()}`,
RULE_ID:id,
DESCRIPTION: help,
})
)
cy.task('table', violationData)
}
Cypress.Commands.add('testAccessibility', (path) => {
cy.visit(path)
cy.injectAxe()
cy.checkA11y(
null,
{
runOnly: {
type: 'tag',
values: ['wcag2a', 'wcag2aa'],
},
},
logViolations,
);
})
This will let us have better results outputting to the console as well as in the Cypress runner. It will clearly highlight and color code issues by impact severity and in the case of the runner, it will also detail all elements affected by each accessibility issue. This is not added to the console since the output gets messed up if you have long texts or too much data. We'll cover later on how we can generate a full report from these results that we can further analyze or even share.
With all this done let's take a look at how our test will look like:
// FILE: cypress/integration/ADA_POC/ada_poc.spec.js
describe('Automated Accessibility Testing POC', ()=>{
it('This page should be accessible.', ()=>{
cy.testAccessibility('');
})
})
As you can see it is now much more readable and simple, and this simple custom command can now be used along any E2E regular automated tests and have the accessibility tested along them. It is worth mentioning that this POC is just using all Axe-core tests on base URL but it can very well be targeted to specific scenarios or even to test after an automated user interaction is performed to see how the accessibility features of the website are behaving in those cases.
But we did a lot of work to get better results and we haven't seen any so far, lets check how the results show up in the console first:
They are looking quite nice and readable to me, at least compared to the previous output we had. Now let's see the same in the Cypress Runner:
Notice that now the issues are clearly listed in the runner details, with colored output and it even lists all the elements that are affected for each type of issue found(those FIXME that you can see in the image). You can even click on the FIXME items and the corresponding html element will be highlighted in the Cypress' runner integrated browser.
The solution now contains the generation of a basic HTML report which includes additional details and resources on the list of issues detected. These help resources are provided for the development team and the test team to get more information about the type of issue and the possible actions that can be taken to correct them. Part of this information provided are the selectors for each of the affected HTML elements.
[FROM AXE-CORE DOCS]
Each rule in axe-core has a number of tags. These provide metadata about the rule. Each rule has one tag that indicates which WCAG version / level it belongs to, or if it doesn't it have the best-practice
tag. If the rule is required by WCAG, there is a tag that references the success criterion number. For example, the wcag111
tag means a rule is required for WCAG 2 success criterion 1.1.1.
The experimental
, ACT
and section508
tags are only added to some rules. Each rule with a section508
tag also has a tag to indicate what requirement in old Section 508 the rule is required by. For example section508.22.a
.
Tag Name | Accessibility Standard / Purpose |
---|---|
wcag2a |
WCAG 2.0 Level A |
wcag2aa |
WCAG 2.0 Level AA |
wcag21a |
WCAG 2.1 Level A |
wcag21aa |
WCAG 2.1 Level AA |
best-practice |
Common accessibility best practices |
wcag*** |
WCAG success criterion e.g. wcag111 maps to SC 1.1.1 |
ACT |
W3C approved Accessibility Conformance Testing rules |
section508 |
Old Section 508 rules |
section508.*.* |
Requirement in old Section 508 |
experimental |
Cutting-edge rules, disabled by default |
cat.* |
Category mappings used by Deque (see below) |
All rules have a cat.*
tag, which indicates what type of content it is part of. The following cat.*
tags exist in axe-core:
Category name |
---|
cat.aria |
cat.color |
cat.forms |
cat.keyboard |
cat.language |
cat.name-role-value |
cat.parsing |
cat.semantics |
cat.sensory-and-visual-cues |
cat.structure |
cat.tables |
cat.text-alternatives |
cat.time-and-media |
📌 Generate a JSON Full Report
📌 Generate an HTML formatted report for sharing [Partially Implemented]
📌 Add example tests of specific elements testing
📌 Add details on which types of tests and rules can be tested with the Axe-Core integration. [Partially Implemented]
That's all for now, let' me think if there is anything else we might need to detail here and I might be back later with more updates for you my README.md friend.
@tzero86