The chapter will focus on important topics to keep your code maintainable in a scaling application. You will learn about code organization to embrace best practices when structuring your folders and files. Another aspect you will learn is testing, which is important to keep your code robust. The whole chapter will take a step back from the practical application and explain a couple of these topics for you.
In JavaScript ES6 you can import and export functionalities from modules. These functionalities can be functions, classes, components, constants and others. Basically everything that you can assign to a variable. The modules can be single files or whole folders with one index file as entry point.
In the beginning of the book, after you have bootstrapped your application with create-react-app, you already had several import
and export
statements across your initial files. Now it is the appropriate time to explain these.
The import
and export
statements help you to share code across multiple files. Before there were already several solutions for this in the JavaScript environment. It was a mess, because you would want to follow one standardized way rather than having several approaches for the same thing. Now it is a native behavior since JavaScript ES6.
Additionally these statements embrace code splitting. You distribute your code across multiple files to keep it reusable and maintainable. The former is true because you can import the piece of code in multiple files. The latter is true because you have one single source where you maintain the piece of code.
Last but not least, it helps you to think about code encapsulation. Not every functionality needs to get exported from a file. Some of these functionalities should only be used in the file where they have been defined. The exports of a file are basically the public API to the file. Only the exported functionalities are available to be reused somewhere else. It follows the best practice of encapsulation.
But let's get practical. How do these import
and export
statements work? The following examples showcase the statements by sharing one or multiple variables across two files. In the end, the approach can scale to multiple files and could share more than simple variables.
You can export one or multiple variables. It is called a named export.
{title="Code Playground: file1.js",lang="javascript"}
const firstname = 'robin';
const lastname = 'wieruch';
export { firstname, lastname };
And import them in another file with a relative path to the first file.
{title="Code Playground: file2.js",lang="javascript"}
import { firstname, lastname } from './file1.js';
console.log(firstname);
// output: robin
You can also import all exported variables from another file as one object.
{title="Code Playground: file2.js",lang="javascript"}
import * as person from './file1.js';
console.log(person.firstname);
// output: robin
Imports can have an alias. It can happen that you import functionalities from multiple files that have the same named export. That's why you can use an alias.
{title="Code Playground: file2.js",lang="javascript"}
import { firstname as foo } from './file1.js';
console.log(foo);
// output: robin
Last but not least there exists the default
statement. It can be used for a few use cases:
- to export and import a single functionality
- to highlight the main functionality of the exported API of a module
- to have a fallback import functionality
{title="Code Playground: file1.js",lang="javascript"}
const robin = {
firstname: 'robin',
lastname: 'wieruch',
};
export default robin;
You can leave out the curly braces for the import to import the default export.
{title="Code Playground: file2.js",lang="javascript"}
import developer from './file1.js';
console.log(developer);
// output: { firstname: 'robin', lastname: 'wieruch' }
Furthermore, the import name can differ from the exported default name. You can also use it in conjunction with the named export and import statements.
{title="Code Playground: file1.js",lang="javascript"}
const firstname = 'robin';
const lastname = 'wieruch';
const person = {
firstname,
lastname,
};
export {
firstname,
lastname,
};
export default person;
{title="Code Playground: file2.js",lang="javascript"}
import developer, { firstname, lastname } from './file1.js';
console.log(developer);
// output: { firstname: 'robin', lastname: 'wieruch' }
console.log(firstname, lastname);
// output: robin wieruch
In named exports you can spare additional lines and export the variables directly.
{title="Code Playground: file1.js",lang="javascript"}
export const firstname = 'robin';
export const lastname = 'wieruch';
These are the main functionalities for ES6 modules. They help you to organize your code, to maintain your code and to design reusable module APIs. You can also export and import functionalities to test them. You will do that in one of the following chapters.
- read more about ES6 import
- read more about ES6 export
You might wonder: Why didn't we follow the best practices of code splitting for the src/App.js file? In the file we already have multiple components which could be defined in their own files/folders (modules). For the sake of learning React, it is practical to keep these things at one place. But once your React application grows, you should consider to split up these components into multiple modules. Only that way your application scales.
In the following, I will propose several module structures you could apply. I would recommend to apply them as an exercise at the end of the book. To keep the book itself simple, I will not perform the code splitting and will continue the following chapters with the src/App.js file.
One possible module structure could be:
{title="Folder Structure",lang="text"}
src/
index.js
index.css
App.js
App.test.js
App.css
Button.js
Button.test.js
Button.css
Table.js
Table.test.js
Table.css
Search.js
Search.test.js
Search.css
It separates the components into their own files, but it doesn't look too promising. You can see a lot of naming duplications and only the file extension differs. Another module structure could be:
{title="Folder Structure",lang="text"}
src/
index.js
index.css
App/
index.js
test.js
index.css
Button/
index.js
test.js
index.css
Table/
index.js
test.js
index.css
Search/
index.js
test.js
index.css
It looks cleaner than before. The index naming of a file describes it as an entry point file to the folder. It is just a common naming convention, but you can use your own naming as well. In this module structure, a component is defined by its component declaration in the JavasScript file, but also by its style and tests.
Another step could be extracting the constant variables from the App component. These constants were used to compose the Hacker News API URL.
{title="Folder Structure",lang="text"}
src/
index.js
index.css
# leanpub-start-insert
constants/
index.js
components/
# leanpub-end-insert
App/
index.js
test.js
index..css
Button/
index.js
test.js
index..css
...
Naturally the modules would split up into src/constants/ and src/components/. Now the src/constants/index.js file could look like the following:
{title="Code Playground: src/constants/index.js",lang="javascript"}
export const DEFAULT_QUERY = 'redux';
export const DEFAULT_HPP = '100';
export const PATH_BASE = 'https://hn.algolia.com/api/v1';
export const PATH_SEARCH = '/search';
export const PARAM_SEARCH = 'query=';
export const PARAM_PAGE = 'page=';
export const PARAM_HPP = 'hitsPerPage=';
The App/index.js file could import these variables in order to use them.
{title="Code Playground: src/components/App/index.js",lang=javascript}
import {
DEFAULT_QUERY,
DEFAULT_HPP,
PATH_BASE,
PATH_SEARCH,
PARAM_SEARCH,
PARAM_PAGE,
PARAM_HPP,
} from '../constants/index.js';
...
When you use the index.js naming convention, you can omit the filename from the relative path.
{title="Code Playground: src/components/App/index.js",lang=javascript}
import {
DEFAULT_QUERY,
DEFAULT_HPP,
PATH_BASE,
PATH_SEARCH,
PARAM_SEARCH,
PARAM_PAGE,
PARAM_HPP,
# leanpub-start-insert
} from '../constants';
# leanpub-end-insert
...
But what's behind the index.js file naming? The convention was introduced in the node.js world. The index file is the entry point to a module. It describes the public API to the module. External modules are only allowed to use the index.js file to import shared code from the module. Consider the following made up module structure to demonstrate it:
{title="Folder Structure",lang="text"}
src/
index.js
App/
index.js
Buttons/
index.js
SubmitButton.js
SaveButton.js
CancelButton.js
The Buttons/ folder has multiple button components defined in its distinct files. Each file can export default
the specific component making it available to Buttons/index.js. The Buttons/index.js file imports all different button representations and exports them as public module API.
{title="Code Playground: src/Buttons/index.js",lang="javascript"}
import SubmitButton from './SubmitButton';
import SaveButton from './SaveButton';
import CancelButton from './CancelButton';
export {
SubmitButton,
SaveButton,
CancelButton,
};
Now the src/App/index.js can import the buttons from the public module API located in the index.js file.
{title="Code Playground: src/App/index.js",lang=javascript}
import {
SubmitButton,
SaveButton,
CancelButton
} from '../Buttons';
By going with this constraint, it would be a bad practice to reach into other files than the index.js in the module. It would break the rules of encapsulation.
{title="Code Playground: src/App/index.js",lang=javascript}
// bad practice, don't do it
import SubmitButton from '../Buttons/SubmitButton';
Now you know how you could refactor your source code in modules with the constraints of encapsulation. As I said, for the sake of keeping the book simple I will not apply these changes. But you should do the refactoring when you finished reading the book.
- refactor your src/App.js file into multiple component modules when you finished the book
The book will not dive deeply into the topic of testing, but it shouldn't be unmentioned. Testing your code in programming is essential and should be seen as mandatory. You want to keep the quality of your code high and an assurance that everything works.
Perhaps you have heard about the testing pyramid. There are end-to-end tests, integration tests and unit tests. If you are not familiar with those, the book gives you a quick and basic overview. A unit test is used to test an isolated and small block of code. It can be a single function that is tested by an unit test. However, sometimes the units work well in isolation yet don't work in combination with other units. They need to be tested as a group as units. That's where integration tests can help out by covering whether units work well together. Last but not least, an end-to-end test is the simulation of a real user scenario. It could be an automated setup in a browser simulating the login flow of an user in a web application. While unit tests are fast and easy to write and to maintain, end-to-end tests are the opposite of this spectrum.
How many tests do I need of each type? You want to have many unit tests to cover your isolated functions. After that, you can have several integration tests to cover that the most important functions work in combination as expected. Last but not least, you might want to have only a few end-to-end tests to simulate critical scenarios in your web application. That's it for the general excursion in the world of testing.
So how do you apply this knowledge in testing your React application? The foundation for testing in React are component tests which can be generalized as unit tests and a part of it as snapshot tests. You will conduct unit tests for your components in the next chapter by using a library called Enzyme. In this chapter, you will focus on another kind of tests: snapshot tests. That's were Jest comes into play.
Jest is a JavaScript testing framework that is used at Facebook. In the React community, it is used for React component tests. Fortunately create-react-app already comes with Jest, so you don't need to worry about setting it up.
Let's start to test your first components. Before you can do that, you have to export the components, which you are going to test, from your src/App.js file. Afterward you can test them in a different file. You have learned about this in the code organization chapter.
{title="src/App.js",lang=javascript}
...
class App extends Component {
...
}
...
export default App;
# leanpub-start-insert
export {
Button,
Search,
Table,
};
# leanpub-end-insert
In your App.test.js file, you will find a first test that came with create-react-app. It verifies that the App component would render without any errors.
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<App />, div);
});
The "it"-block describes one test case. It comes with a test description and when you test it, it can either succeed or fail. Furthermore, you could wrap it into a "describe"-block that defines your test suit. A test suit could include a bunch of the "it"-blocks for one specific component. You will see those "describe"-blocks later on. Both blocks are used to separated and organize your test cases.
You can run your test cases by using the interactive create-react-app test script on the command line. You will get the output for all test cases on your command line interface.
{title="Command Line",lang="text"}
npm test
Note: If errors show up when you run the single test for the App component for the first time, it could be because of the unsupported fetch method that is used in fetchSearchTopStories()
which is triggered in componentDidMount()
. You can make it work by following these two steps:
- On the command line, install the following package:
npm install isomorphic-fetch
- Include it in your App.js file:
import fetch from 'isomorphic-fetch';
Now Jest enables you to write snapshot tests. These tests make a snapshot of your rendered component and run this snapshot against future snapshots. When a future snapshot changes, you will get notified in the test. You can either accept the snapshot change, because you changed the component implementation on purpose, or deny the change and investigate for the error. It complements unit tests very well, because you only test the diffs of the rendered output. It doesn't add big maintenance costs, because you can simply accept changed snapshots when you changed something on purpose for the rendered output in your component.
Jest stores the snapshots in a folder. Only that way it can validate the diff against a future snapshot. Additionally, the snapshots can be shared across teams by having them in one folder.
Before writing your first snapshot test with Jest, you have to install an utility library.
{title="Command Line",lang="text"}
npm install --save-dev react-test-renderer
Now you can extend the App component test with your first snapshot test. First, import the new functionality from the node package and wrap your previous "it"-block for the App component into a descriptive "describe"-block. In this case, the test suit is only for the App component.
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
# leanpub-start-insert
import renderer from 'react-test-renderer';
# leanpub-end-insert
import App from './App';
# leanpub-start-insert
describe('App', () => {
# leanpub-end-insert
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<App />, div);
});
# leanpub-start-insert
});
# leanpub-end-insert
Now you can implement your first snapshot test by using a "test"-block.
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import renderer from 'react-test-renderer';
import App from './App';
describe('App', () => {
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<App />, div);
});
# leanpub-start-insert
test('has a valid snapshot', () => {
const component = renderer.create(
<App />
);
let tree = component.toJSON();
expect(tree).toMatchSnapshot();
});
# leanpub-end-insert
});
Run your tests again and see how the tests either succeed or fail. They should succeed. Once you change the output of the render block in your App component, the snapshot test should fail. Then you can decide to update the snapshot or investigate in your App component.
Basically the renderer.create()
function creates a snapshot of your App component. It renders it virtually and stores the DOM into a snapshot. Afterward, the snapshot is expected to match the previous snapshot from when you ran your snapshot tests the last time. This way, you can assure that your DOM stays the same and doesn't change anything by accident.
Let's add more tests for our independent components. First, the Search component:
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import renderer from 'react-test-renderer';
# leanpub-start-insert
import App, { Search } from './App';
# leanpub-end-insert
...
# leanpub-start-insert
describe('Search', () => {
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<Search>Search</Search>, div);
});
test('has a valid snapshot', () => {
const component = renderer.create(
<Search>Search</Search>
);
let tree = component.toJSON();
expect(tree).toMatchSnapshot();
});
});
# leanpub-end-insert
The Search component has two tests similar to the App component. The first test simply renders the Search component to the DOM and verifies that there is no error during the rendering process. If there would be an error, the test would break even though there isn't any assertion (e.g. expect, match, equal) in the test block. The second snapshot test is used to store a snapshot of the rendered component and to run it against a previous snapshot. It fails when the snapshot has changed.
Second, you can test the Button component whereas the same test rules as in the Search component apply.
{title="src/App.test.js",lang=javascript}
...
# leanpub-start-insert
import App, { Search, Button } from './App';
# leanpub-end-insert
...
# leanpub-start-insert
describe('Button', () => {
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<Button>Give Me More</Button>, div);
});
test('has a valid snapshot', () => {
const component = renderer.create(
<Button>Give Me More</Button>
);
let tree = component.toJSON();
expect(tree).toMatchSnapshot();
});
});
# leanpub-end-insert
Last but not least, the Table component that you can pass a bunch of initial props to render it with a sample list.
{title="src/App.test.js",lang=javascript}
...
# leanpub-start-insert
import App, { Search, Button, Table } from './App';
# leanpub-end-insert
...
# leanpub-start-insert
describe('Table', () => {
const props = {
list: [
{ title: '1', author: '1', num_comments: 1, points: 2, objectID: 'y' },
{ title: '2', author: '2', num_comments: 1, points: 2, objectID: 'z' },
],
};
it('renders without crashing', () => {
const div = document.createElement('div');
ReactDOM.render(<Table { ...props } />, div);
});
test('has a valid snapshot', () => {
const component = renderer.create(
<Table { ...props } />
);
let tree = component.toJSON();
expect(tree).toMatchSnapshot();
});
});
# leanpub-end-insert
Snapshot tests usually stay pretty basic. You only want to cover that the component doesn't change its output. Once it changes the output, you have to decide if you accept the changes. Otherwise you have to fix the component when the output is not the desired output.
- see how a snapshot test fails once you change your component's return value in the
render()
method- either accept or deny the snapshot change
- keep your snapshots tests up to date when the implementation of components change in next chapters
- read more about Jest in React
Enzyme is a testing utility by Airbnb to assert, manipulate and traverse your React components. You can use it to conduct unit tests to complement your snapshot tests in React.
Let's see how you can use enzyme. First you have to install it since it doesn't come by default with create-react-app. It comes also with an extension to use it in React.
{title="Command Line",lang="text"}
npm install --save-dev enzyme react-addons-test-utils enzyme-adapter-react-16
Second, you need to include it in your test setup and initialize its Adapter for using it in React.
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import renderer from 'react-test-renderer';
# leanpub-start-insert
import Enzyme from 'enzyme';
import Adapter from 'enzyme-adapter-react-16';
# leanpub-end-insert
import App, { Search, Button, Table } from './App';
# leanpub-start-insert
Enzyme.configure({ adapter: new Adapter() });
# leanpub-end-insert
Now you can write your first unit test in the Table "describe"-block. You will use shallow()
to render your component and assert that the Table has two items, because you pass it two list items. The assertion simply checks if the element has two elements with the class table-row
.
{title="src/App.test.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import renderer from 'react-test-renderer';
# leanpub-start-insert
import Enzyme, { shallow } from 'enzyme';
# leanpub-end-insert
import Adapter from 'enzyme-adapter-react-16';
import App, { Search, Button, Table } from './App';
...
describe('Table', () => {
const props = {
list: [
{ title: '1', author: '1', num_comments: 1, points: 2, objectID: 'y' },
{ title: '2', author: '2', num_comments: 1, points: 2, objectID: 'z' },
],
};
...
# leanpub-start-insert
it('shows two items in list', () => {
const element = shallow(
<Table { ...props } />
);
expect(element.find('.table-row').length).toBe(2);
});
# leanpub-end-insert
});
Shallow renders the component without its child components. That way, you can make the test very dedicated to one component.
Enzyme has overall three rendering mechanisms in its API. You already know shallow()
, but there also exist mount()
and render()
. Both instantiate instances of the parent component and all child components. Additionally mount()
gives you access to the component lifecycle methods. But when to use which render mechanism? Here some rules of thumb:
- Always begin with a shallow test
- If
componentDidMount()
orcomponentDidUpdate()
should be tested, usemount()
- If you want to test component lifecycle and children behavior, use
mount()
- If you want to test a component's children rendering with less overhead than
mount()
and you are not interested in lifecycle methods, userender()
You could continue to unit test your components. But make sure to keep the tests simple and maintainable. Otherwise you will have to refactor them once you change your components. That's why Facebook introduced snapshot tests with Jest in the first place.
- write a unit test with Enzyme for your Button component
- keep your unit tests up to date during the following chapters
- read more about enzyme and its rendering API
You may know TypeScript or Flow to introduce a type interface to JavaScript. A typed language is less error prone, because the code gets validated based on its program text. Editors and other utilities can catch these errors before the program runs. It makes your program more robust.
In the book, you will not introduce Flow or TypeScript, but another neat way to check your types in components. React comes with a built-in type checker to prevent bugs. You can use PropTypes to describe your component interface. All the props that get passed from a parent component to a child component get validated based on the PropTypes interface assigned to the child component.
The chapter will show you how you can make all your components type safe with PropTypes. I will omit the changes for the following chapters, because they add unnecessary code refactorings. But you should keep and update them along the way to keep your components interface type safe.
First, you have to install a separate package for React.
{title="Command Line",lang="text"}
npm install prop-types
Now, you can import the PropTypes.
{title="src/App.js",lang=javascript}
# leanpub-start-insert
import PropTypes from 'prop-types';
# leanpub-end-insert
Let's start to assign a props interface to the components:
{title="src/App.js",lang=javascript}
const Button = ({ onClick, className = '', children }) =>
<button
onClick={onClick}
className={className}
type="button"
>
{children}
</button>
# leanpub-start-insert
Button.propTypes = {
onClick: PropTypes.func,
className: PropTypes.string,
children: PropTypes.node,
};
# leanpub-end-insert
Basically that's it. You take every argument from the function signature and assign a PropType to it. The basic PropTypes for primitives and complex objects are:
- PropTypes.array
- PropTypes.bool
- PropTypes.func
- PropTypes.number
- PropTypes.object
- PropTypes.string
Additionally you have two more PropTypes to define a renderable fragment (node), e.g. a string, and a React element:
- PropTypes.node
- PropTypes.element
You already used the node
PropType for the Button component. Overall there are more PropType definitions that you can read up in the official React documentation.
At the moment all of the defined PropTypes for the Button are optional. The parameters can be null or undefined. But for several props you want to enforce that they are defined. You can make it a requirement that these props are passed to the component.
{title="src/App.js",lang=javascript}
Button.propTypes = {
# leanpub-start-insert
onClick: PropTypes.func.isRequired,
# leanpub-end-insert
className: PropTypes.string,
# leanpub-start-insert
children: PropTypes.node.isRequired,
# leanpub-end-insert
};
The className
is not required, because it can default to an empty string. Next you will define a PropType interface for the Table component:
{title="src/App.js",lang=javascript}
# leanpub-start-insert
Table.propTypes = {
list: PropTypes.array.isRequired,
onDismiss: PropTypes.func.isRequired,
};
# leanpub-end-insert
You can define the content of an array PropType more explicit:
{title="src/App.js",lang=javascript}
Table.propTypes = {
list: PropTypes.arrayOf(
PropTypes.shape({
objectID: PropTypes.string.isRequired,
author: PropTypes.string,
url: PropTypes.string,
num_comments: PropTypes.number,
points: PropTypes.number,
})
).isRequired,
onDismiss: PropTypes.func.isRequired,
};
Only the objectID
is required, because you know that some of your code depends on it. The other properties are only displayed, thus they are not necessarily required. Moreover you cannot be sure that the Hacker News API has always a defined property for each object in the array.
That's it for PropTypes. But there is one more aspect. You can define default props in your component. Let's take again the Button component. The className
property has an ES6 default parameter in the component signature.
{title="src/App.js",lang=javascript}
const Button = ({
onClick,
className = '',
children
}) =>
...
You could replace it with the internal React default prop:
{title="src/App.js",lang=javascript}
# leanpub-start-insert
const Button = ({
onClick,
className,
children
}) =>
# leanpub-end-insert
<button
onClick={onClick}
className={className}
type="button"
>
{children}
</button>
# leanpub-start-insert
Button.defaultProps = {
className: '',
};
# leanpub-end-insert
Same as the ES6 default parameter, the default prop ensures that the property is set to a default value when the parent component didn't specify it. The PropType type check happens after the default prop is evaluated.
If you run your tests again, you might see PropType errors for your components on your command line. It can happen because you didn't define all props for your components in the tests that are defined as required in your PropType definition. The tests themselves all pass correctly though. You can pass all required props to avoid these errors.
- define the PropType interface for the Search component
- add and update the PropType interfaces when you add and update components in the next chapters
- read more about React PropTypes
{pagebreak}
You have learned how to organize your code and how to test it! Let's recap the last chapters:
- React
- PropTypes let you define type checks for components
- Jest allows you to write snapshot tests for your components
- Enzyme allows you to write unit tests for your components
- ES6
- import and export statements help you to organize your code
- General
- code organization allows you to scale your application with best practices
You can find the source code in the official repository.