Настройка CI/CD пайплайна для тестирования доступности
Автоматизированная проверка доступности в CI/CD предотвращает регрессии: каждый PR проходит аудит, и если появляется новое нарушение WCAG — сборка падает. Комбинируют несколько инструментов: axe-core для компонентного тестирования, Playwright для E2E, Lighthouse CI для страничного аудита.
Стратегия покрытия
Unit-тесты компонентов (Jest + jest-axe)
→ 100% React-компонентов через Storybook
→ Запуск на каждый коммит
E2E-тесты страниц (Playwright + axe-core)
→ Ключевые страницы: главная, каталог, форма, checkout
→ Запуск на PR в main
Lighthouse CI (страничный аудит)
→ Бюджет: минимум 90/100 по accessibility
→ Запуск на PR в main
Еженедельный полный аудит (Pa11y + sitemap)
→ Все публичные страницы
→ Отчёт в Slack
GitHub Actions: полный пайплайн
# .github/workflows/accessibility.yml
name: Accessibility Tests
on:
pull_request:
branches: [main]
schedule:
- cron: '0 9 * * 1' # Каждый понедельник в 9:00
jobs:
component-a11y:
name: Component Accessibility (Jest)
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with: { node-version: '20' }
- run: npm ci
- run: npm test -- --testPathPattern="a11y|accessibility" --coverage=false
env:
CI: true
e2e-a11y:
name: E2E Accessibility (Playwright)
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with: { node-version: '20' }
- run: npm ci
- run: npx playwright install chromium --with-deps
- name: Start app
run: npm run build && npm start &
- name: Wait for app
run: npx wait-on http://localhost:3000 --timeout 60000
- name: Run accessibility tests
run: npx playwright test tests/accessibility/
- uses: actions/upload-artifact@v4
if: failure()
with:
name: playwright-report
path: playwright-report/
lighthouse-a11y:
name: Lighthouse Audit
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with: { node-version: '20' }
- run: npm ci && npm run build
- name: Run Lighthouse CI
uses: treosh/lighthouse-ci-action@v10
with:
urls: |
http://localhost:3000
http://localhost:3000/catalog
budgetPath: .lighthouserc.json
temporaryPublicStorage: true
env:
LHCI_GITHUB_APP_TOKEN: ${{ secrets.LHCI_GITHUB_APP_TOKEN }}
Playwright: тест доступности страниц
// tests/accessibility/pages.spec.ts
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
const PAGES_TO_TEST = [
{ path: '/', name: 'Главная' },
{ path: '/catalog', name: 'Каталог' },
{ path: '/contact', name: 'Контакты' },
];
for (const { path, name } of PAGES_TO_TEST) {
test(`${name}: WCAG 2.1 AA`, async ({ page }) => {
await page.goto(path);
await page.waitForLoadState('networkidle');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa'])
.analyze();
// Детальный вывод нарушений
if (results.violations.length > 0) {
const summary = results.violations.map(v =>
`[${v.impact}] ${v.id}: ${v.description} (${v.nodes.length} элементов)`
).join('\n');
console.error(`Нарушения на "${name}":\n${summary}`);
}
expect(results.violations).toHaveLength(0);
});
}
Репортинг в Pull Request
# Комментарий к PR с результатами аудита
- name: Comment PR
uses: thollander/actions-comment-pull-request@v2
if: always()
with:
message: |
## Отчёт доступности
| Инструмент | Статус |
|---|---|
| Jest + axe | ${{ job.status == 'success' && '✅ Пройден' || '❌ Ошибки' }} |
| Playwright | ${{ needs.e2e-a11y.result == 'success' && '✅ Пройден' || '❌ Ошибки' }} |
| Lighthouse | ${{ needs.lighthouse-a11y.result == 'success' && '✅ ≥ 90' || '❌ < 90' }} |
Известные исключения
Фиксируем ложноположительные срабатывания в конфиге:
// axe-config.js — исключения для третьесторонних виджетов
const AXE_CONFIG = {
rules: [
// Виджет чата (третья сторона, не контролируем)
{ id: 'color-contrast', selector: '#intercom-frame', enabled: false },
],
};
Сроки
Полный CI/CD пайплайн доступности с Jest, Playwright, Lighthouse и отчётом в PR: 3–4 рабочих дня.







