All types of testing (Unit, Integration, E2E) are supported in Application.
- Unit + Integration testing is covered by Jest (Testing Framework) and React Testing Library (Testing Util);
💡 Note: A11y testing is also supported on Unit/Integration level and covered by Jest-Axe.
- E2E testing is covered by Cypress and extended by Cypress Testing Library;
💡 Note: A11y testing is also supported on E2E level and covered by Cypress-Axe.
Unit/Integration testing is added to CI/CD pipeline as a job:
- for feature branches Test job only runs existing Unit and Integration tests failing the pipeline in case any test was not passed;
💡 Note: Follow
test-unit-integration
job of CircleCI config for more details. - for main branch Test job also includes Reports generation (Results + Code Coverage). This reports than are used to visualize according test data in convenient way (using CircleCI Test Insights for displaying test results and CodeCov to display code coverage report);
💡 Note: Follow
test-unit-integration-with-reports
job of CircleCI config for more details.
Performance testing is also added to CI/CD pipeline as a job. It only runs on main branch.
💡 Note: Follow
test-performance
job of CircleCI config for more details.
Info: Unit Testing is to test each part of the program and show that the individual parts (modules, functions, components) are correct, while Integration Testing is to combine such parts and test as a group to see that their combination working fine.
Unit/Integration Tests config - (/test/jest) - used for storing Unit/Integration Tests framework configuration (Jest config) and custom testing utils setup (e.g. RTL custom utils);
For convenience purpose:
- all Unit/Integration (+A11y) testing utilities provided from RTL/Axe are stored and could be exported from
'@test-utils'
module (allias for /config/test/jest/test-utils/); - all custom testing utilities (e.g. render decorated with Providers) are stored and could be exported from
'@test-utils/custom'
module (allias for /config/test/jest/test-utils/custom/);
- follow F.I.R.S.T. principle;
💡 Note: Fast - each test should run and show you the desired output in a matter of seconds; Isolated - each test should be independent of everything else so that it results is not influenced by any other factor; Repeateble - each test should be repeatable and deterministic, it's values shouldn’t change based on being run on different environments. Unit Tests should own their data and not depend on any external factors (side-effects); Self-validating - each test should provide readable result whether it is passed or failed, you should not do it manualy; Thorough - tests should cover as much scenarios as possible(happy path, edge cases, security). Strive for it, but remember about next convention (see below).
- do not write test "just for test", remember there are no such thing as "ideal code coverage". Strive to the best coverage, but do it reasonably;
- Unit/Integration tests should be added in separate sub-folder (__tests__);
- test file name should follow template:
[name].spec.{js|ts|tsx}
; - tests should only verify single file (do NOT verify several components/modules/services scoped in different files in scope of single test file);
- if tests require some solid mocking:
- it should be stored in scope of (/__tests__) sub-folder in separate file;
- mock files should follow template:
[name].mock.{js|ts}
;
describe
andtest
(it
) blocks naming:- strive to follow Gherkin style (Given-When-Then) as it helps to stricture your test files and increase their readability;
- the high level
describe
block should be named as an entity (function, component, etc.) that it tests; - in case tested file includes multiple exports to test - write
describe
block per each exported entity; - start
describe
block with "given..." or "when..." (following Gherkin style); - start
test
(it
) block with "should..." or "then...";💡 Note: As a result your test scenario will be readable, e.g.:
"given USER sees the button"
->"when USER clicked on button"
->"then 'Hooray!' message became shown"
- use test hooks (
beforeEach
/beforeAll
/afterEach
/afterAll
) following their purpose: pre-test configuration (before*
) OR cleaning up (after*
);
- follow General rules;
- if Component should be covered with general Unit/Integration tests:
- tests should be stored in scope (/__tests__) sub-folder in separate file;
- general Test file name should follow template:
[name].component.spec.{js|tsx}
;
- if Component should be covered with A11y Tests:
- tests should be stored in scope (/__tests__) sub-folder in separate file;
- A11y Test file name should follow template:
[name].component.a11y.spec.{js|tsx}
;
- try to avoid Snapshot testing (using it could lead to possible false negative/false positive test results + it is usually slow);
- React Testing Library rules:
-
focus on USER behavior, NOT on implementation details (RTL stricts access to component's internals (e.g. state));
-
there are 3 common variants of element querying in RTL:
get*
,find*
,query*
. Generally please follow simple rule of usage (more details):Use
get*
as a default option -> if need to identify an element is NOT presented - usequery*
(get*
will throw an error in such case) -> if need handle the scenario with a result of an asynchronous action - usefind*
-
focus on correct ("jest-dom") assertions (more details)
❌
expect(button.disabled).toBe(true)
-> ✅expect(button).toBeDisabled()
Some of available matcher methods include:
.toBeInTheDocument()
,.toBeVisible()
,.toHaveValue()
,.toHaveStyle()
,.toBeDisabled()
, etc. -
use "user-event" in priority of "fire-event" where possible (more details);
-
try to avoid using
container
to query for elements (use "screen") (more details) -
covering React custom Hooks:
- use
renderHook
function of @testing-library/react for covering React custom Hook; - do NOT create helping Functional Components;
- cover only Custom hooks which are reusable or potentially reusable, other custom Hooks could be covered in scope of specific Component usage;
- use
-
Info: Nice article by Kent C. Dodds (founder of Testing Library): https://kentcdodds.com/blog/common-mistakes-with-react-testing-library
Info: E2E(End-to-End) Testing is a technique that tests the entire software product from the beginning to the end from USER perspective. It ensures the application flow behaves as expected by simulating the real USER scenario.
E2E Tests config - (/test/cypress) - used for storing E2E Tests framework configuration (Cypress config) and E2E test cases; It consists of:
- E2E Common Tests config - Common E2E tests configuration;
- E2E A11y Tests config - A11y E2E tests configuration;
- mind that E2E tests are usually the most expensive and time-consuming test type, they are at the top of Testing Pyramid, so for each scenario:
- let down your tests as deep as possible in Testing Pyramid, in case some scenario could be covered with Unit/Integration test - do it there;
- do NOT duplicate your test cases between different Testing Pyramid layers;
describe
andtest
(it
) blocks naming:- strive to follow Gherkin style (Given-When-Then) as it helps to stricture your test files and increase their readability;
- the high level
describe
block should decribe the whole scenario that it tests; - start
describe
block with "given..." or "when..." (following Gherkin style); - start
test
(it
) block with "should..." or "then...";💡 Note: As a result your test scenario will be readable, e.g.:
"given USER comes to some page"
->"when USER clicked on button"
->"then 'Hooray!' message became shown"
- if new general E2E test should be added:
- test files should be stored in scope of /config/test/cypress/e2e folder;
- test file name should follow template:
[name].cy.js
;
- if new A11y E2E test should be added:
- test files should be stored in scope of /config/test/cypress/e2e/a11y folder;
- test file name should follow template:
[name].a11y.cy.js
;
Performance & Insights Testing is only supported on CI/CD level and handled by LightHouse CI. Configuration file could be found here: /config/test/lighthouse.config.js.
💡 Note: PWA Insights testing right now is turned OFF and NOT a part of general flow because PWA supported only as separate independent build type.
💡 Note: For manual Performance Testing you can use Chrome built-in Lighthouse DevTool.