React and React Native components cannot be treated in the same way when it comes to testing. React is for creating web apps that run in a web browser. React Native is for creating Native mobile applications that run directly on an android or iOS device. So while there are many similarities, there are also some core differences when it comes to testing.
For testing React and React Native components you could use any testing framework such as mocha, jasmine or tape. Facebook have created their own framework especially for React called jest. It is based on jasmine and this is the framework used throughout this guide.
If we consider the component only as a view with as little logic in it as possible, this will greatly simplify our testing efforts. This is how the app should be designed, especially considering the effort you will have to go for getting React Native testing right.
React is much easier to get up and running with in terms of testing than it's native counterpart. Much of the work around tooling has been done by facebook, and airbnb. All you need to do is simulate a browser dom in the global environment of your node process using a package like jsdom. Used in conjuction with enzyme, a library that allows you to interact with React components from a user perspective, you have pretty much all your testing needs covered.
A bit of setup is required to get everything working correctly.
npm install jest enzyme babel-jest babel-preset-es2015 babel-preset-react react-test-renderer
- Create a .babelrc file with the following if you haven't already:
{ "presets": ["es2015", "react"] }
That's it, you're ready to test!
We can use Enzyme's shallow function to render only the current component and its children upto one level deep.
Take a look at the shallow native example.
We can also use Snapshot testing, if the component is simple enough. In some cases this will remove the need for any other. We cover this below.
For integration testing we can use Enzyme's mount function. We don't need to do any more setup if using jest.
Check out the web mount example
In my opinion this is the only thing we really need when it comes to integration testing in React. We will want to stub and mock out some components.
Most functional testing should be done with the mount function and snapshot testing.
Some of it could be done with webdriver. Webdriver testing is well established and doesn't need to be explained here. But unless you need visual validation, it will be a bit superflous.
Snapshot testing ensures that components do not change unexpectedly. It's not a testing approach you can apply in a TDD way, but it is useful in catching unexpexted changes.
The first time you run a snapshot test it will create and save a snapshot of the component you are testing. On subsequent runs it will compare the saved snapshot to a freshly generated one. If it has changed the test will fail and you will need to approve the new snapshot.
Check out the web snapshot test example
Change the value of the props components in the test, rerun the test, and see what happens.
Testing React Native components is a bit like dealing with jekyll and hyde. Unit testing and snapshot testing are fairly straightforward. Integration and functional testing require a lot more setup.
First, a little setup:
In package.json, add the following:
...
"jest": {
"preset": "react-native",
"setupTestFrameworkScriptFile": "./node_modules/jest-enzyme/lib/index.js"
}
...
We can use Enzyme's shallow function to render only the current component and its children upto one level deep.
Take a look at the shallow native example.
We can also use Snapshot testing, if the component is simple enough. In some cases this will remove the need for any other
For integration testing we can use Enzyme's mount function. To get this working requires some fiddling as Enzyme is designed for web component testing. Fortunately for us, most of this work has been done with react-native-mock-render. This is still a work in progress, which means that mount doesn't work as flawlessly as it does with React web components.
The following setup is required:
npm install -D react-native-mock-render jsdom
- Create a test setup file
- Add a reference to the setup file in your jest config in package.json:
"jest": {
...
"setupFiles": [
"./tests/test-setup.js"
]
...
}
Then you can write an integration test using mount, check out the mount native example
For integration testing, it is also useful to mock out components that you are not interested in. Jest makes this easy to do.
Functional testing can be performed in one of several ways.
Using Enzyme's mount function, snapshot testing, or through UI automation with a tool such as appium, cavy, or native testing tools such as espresso and XCTest.
The mount function and snapshot testing have one major drawback for ReactNative: they do not run within an environment that is really comparable to the system that the component under test will eventually run on: a mobile OS. For example, one issue is that when mounting a component it seems that twice as many components are 'found' than should be.
From what I have read, the most prominent issue with React components is the differences in how they behave on device between screen sizes and platforms. Therefore, at least a small degree of functional testing must be done on the device itself. A React component will be the interface to the user. It is important that components are tested from this perspective.
On the other hand, testing on device is slow. Especially with appium. It is faster with native tools, but that doubles the development overhead. Cavy is a neat solution, but also means that you have to add refs to your components specifically for testing.
Check out this Cavy example:
Cavy is extremely powerful, and quick as it allows you direct access to the React-Native runtime within a device. The sample test runs the tests, but you do not actually see the visual rendering. However cavy can still inspect what is being rendered in the React-Native runtime - which is awesome. This could possibly run at similar speeds to api integration tests.
I'll add a description of what is actually going on soon.
It works in the same way that it does for web components created with React.
Check out the native snapshot test example
Change the value of the props components in the test, rerun the test, and see what happens.
Property based testing is a generative testing method. You define properties for your functions, and you define a input generators that will generate randomised inputs within a set of constraints. Then, tests will be created that will verify that each input is valid for the given property. In this way thousands of tests can be generated. It is a good way of catching edge cases that a developer may miss when writing example driven tests.
Property based testing comes from the functional programming paradigm. Quickcheck, a property based testing library developed for haskell, has inspired many implementations in other languages, including javascript.
There are two main implementations in javascript: jsverify, and testcheck. I prefer testcheck as it is simpler to use and easier to read, for me at least.
You can check out the property based testing example.
We have a simple component which takes an input property and squares it. Our test checks that the property, that all displayed values are greater than or equal to zero, holds for a range of values between -infinity and inifinity.
This test uncovered an input that I hadn't accounted for: NaN.
It is possible to generate far more complex inputs with the input generator. Check out the docs for more details.
This is another generative testing method. Not popularised except in safety critical industries and no javascript implementations of this exist. It may be useful for react (and even more general) testing. I will draw up a simple POC of it with a larger React component.
references: