Speaking about UI Testing (remember that we are speaking about the UI only, not the underlying JavaScript code), there are three main test types:
-
Component tests: the unit tests of a UI, they test every single component in an isolated environment.
Developing components in isolation is important because it allows you to isolate them from the corresponding container/use. A component exists to isolate a single behavior/content (the Single responsibility principle) and therefore, coding it in isolation is profitable.
There are many ways and tools to develop components in isolation but Storybook became the standard choice because of its effectiveness and its ecosystem.
Components have three types of contracts: the generated output (HTML), their visual aspect (CSS), and the external APIs (props and callbacks). Testing every aspect could be cumbersome, that's where Storyshots comes in handy. It allows you to automate:
-
the snapshot tests: a snapshot is an output generated by your component, a string containing all the rendered HTML. If the generated HTML changes, accidentally or not, the snapshot tests fail and you can choose if the changes were intentional or not.
-
the visual regression tests: the visual aspect of the component compared pixel by pixel with the previous one, again, you are prompted to choose if you accept the changes or not.
These tests are launched by Storyshots automatically on every Storybook page (AKA "stories").
-
the callback tests: a small React container app renders the component passing it some callbacks. Then, the user interactions are simulated and passed the callback is expected to be called. React Testing Library is the standard library of choice for this kind of tests
-
the interaction/state tests: some interactions with a component expect correct state management. This kind of test must be written from the consumer point of view, not from the inner one (ex. the value of the input field when the user fills it, not the inner component state). An interaction/state test should assert the input field value after the keyboard events triggered.
-
Alternatively, Cypress came out with its own solution in order to launch component tests iside it, take a look at the Unit Testing React components with Cypress chapter.
-
UI integration tests: they run the whole app in a real browser without hitting a real server. These tests are the ace in the hole of every front-end developer. They are blazing fast and less exposed to random failures or false negatives.
The front-end application does not know that there is not a real server: every AJAX call is resolved in no time by the testing tool. Static JSON responses (called "fixtures") are used to simulate the server responses. Fixtures allow us to test the front-end state simulating every possible back-end state.
Another interesting effect: Fixtures allow you to work without a working back-end application. You can think about UI integration tests as "front-end-only tests".
At the core of the most successful test suites, there is a lot of UI integration tests, considering the best type of test for your front-end app.
-
End-to-end (E2E) tests: they run the whole app interacting with the real server. From the user interactions (one of the "end") to the business data (the other "end"): everything must work as designed. E2E tests are typically slow because
- they need a working back-end application, typically launched alongside the front-end application. You can't launch them without a server, so you depend on the back-end developers to work
- they need reliable data, seeding and cleaning it before every test
That's why E2E tests are not feasible to be used as the only/main test type. They are pretty important because they are testing everything (front-end + back-end) but they must be used carefully to avoid brittle and hour-long test suites.
In a complete suite with a lot of UI integration tests, you can think about E2E tests as "back-end tests". What flows should you test through them?
- the Happy Path flows: you need to be sure that, at least, the users are able to complete the basic operations
- everything valuable for your business: happy path or not, test whatever your business cares about (prioritizing them, obviously)
- everything that breaks often: weak areas of the system must be monitored too
Identifying/defining the type of test is useful to group them, to name the test files, to limit their scope, and to choose where to run them or not though the whole application and deployment pipelines.