Pomodoro And Jest
Pomodoro doesn't use Enzyme, since the Enzyme library is discontinued; the official replacement for it is React Testing Library.
Installation
There are a few dependencies that have to be installed for Jest and React Testing Library:
{
"devDependencies": {
"@testing-library/dom": "^8.20.1",
"@testing-library/jest-dom": "^6.6.3",
"@testing-library/react": "^12.1.5",
"@testing-library/react-hooks": "^8.0.1",
"@types/jest": "^29.5.11",
"@types/redux-mock-store": "^1.0.4",
"jest": "^29.7.0",
"jest-environment-jsdom": "^29.7.0"
}
}
Since Pomodoro is Webpack-based, it is required to have paths resolving added to its configuration:
module.exports = {
resolve: {
extensions: ['.js', '.jsx', '.ts', '.tsx', '.json'],
alias: {
'react-dom': IS_DEV
?
'@hot-loader/react-dom': 'react-dom',
'@/*': path.resolve(__dirname,
'../src/*'
)
}
}
}
Also, the same for Jest configuration, along with some mocks for files:
{
moduleNameMapper: {
'\\.(css|less|sass|scss)$': 'identity-obj-proxy',
'\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$':
'<rootDir>/src/scripts/fileMock.js',
'^@/(.*)$': ['<rootDir>/src/$1']
}
}
A Shallow Test
Pomodoro uses Redux to store much of the application data; however, there are a number of plain components using properties. Tests are just checking these components can be rendered. The same can be checked for more complex components:
import React from 'react';
import { render } from '@testing-library/react';
import '@testing-library/jest-dom';
import { FocusTotals } from './FocusTotals';
import styles from './focustotals.css';
describe('FocusTotals', () => {
it('renders without crashing', () => {
const {container} = render(<FocusTotals />);
expect(container).toBeInTheDocument();
});
})
The key function is render
which writes the component being tested to virtual
DOM returning the object which can be used in the tests.
Testing Styles
Components are being decorated differently, depending on their data, and this can be validated in tests, for example:
it('applies active class when percent is greater than 0', () => {
const { container } = render(<FocusTotals percent={50} />);
expect(container.firstChild).toHaveClass(styles.active);
});
it('does not apply active class when percent is 0 or less', () => {
const { container } = render(<FocusTotals percent={0} />);
expect(container.firstChild).not.toHaveClass(styles.active);
});
Dynamic Text Correctness
For text which is only visible on runtime, it is useful to create a simple test to check some cases:
const mockDay: IDayStats = {
name: "Monday",
date: new Date("2023-10-10"),
time: 120,
pause: 10,
break: 5,
stops: 2,
short: '1',
};
it("should render the day name and date correctly", () => {
const { getByText } = render(<DayTotals day={mockDay} time="120" />);
expect(getByText("Monday (10/10/2023)")).toBeInTheDocument();
});
User Interaction With A Form
Pomodoro uses @testing-library/react
to fire user clicks and do some
typing in the input fields of components being tested:
import { render, screen, fireEvent } from '@testing-library/react';
test('shows error message if task name is less than 3 characters', () => {
render(<AddTaskForm onSubmit={jest.fn()} />);
const input = screen.getByPlaceholderText('Название задачи');
const button = screen.getByText('Добавить');
fireEvent.change(input, { target: { value: 'ab' } });
fireEvent.click(button);
expect(screen.getByText('Введите не меньше трех символов для новой задачи')).toBeInTheDocument();
});
In the case, the above Jest assumes 'ab' should trigger an error message and validates the form for the presence of the message.
Testing With Redux Store
Some components are using data from Redux store. To test the components along with this data, a new Redux Mock store can be created:
import configureStore from 'redux-mock-store';
const mockStore = configureStore([]);
describe('WeekChart', () => {
let store: ReturnType<typeof mockStore>;
let initialState: RootState;
})
Pomodoro project is yet to be upgraded to use the recommended Redux Toolkit. So this approach is deprecated and will be rewritten.
Despite the deprecation of the mock store, the main idea will work with a real store in RTK tests too.
First, initial data for tests is created. Note that the data should be for the previous week related to tests' date:
beforeEach(() => {
const monday = getPreviousMonday(new Date());
initialState = {
day: {
date: new Date('2023-10-01'),
short: 'Sun',
time: 0,
pause: 0,
break: 0,
stops: 0,
name: '',
} as IDayStats,
week: {
start: monday,
end: getAnotherDay(monday, 7),
id: '1',
text: '',
} as IWeek,
stats: [
{
date: getAnotherDay(monday, -1),
short: 'Sun',
time: 120,
pause: 0,
break: 0,
stops: 0,
name: 'Воскресенье',
},
{
date: monday,
short: 'Пн',
time: 150,
pause: 0,
break: 0,
stops: 0,
name: 'Понедельник',
}
//...
]
}
})
So the next test is to verify the bar chart in the component is rendered with correct vertical line tick labels:
it('renders horizontal lines with correct times', () => {
render(
<Provider store={store}>
<WeekChart />
</Provider>,
);
expect(screen.getByText('42 сек')).toBeInTheDocument();
expect(screen.getByText('1 мин 24 сек')).toBeInTheDocument();
expect(screen.getByText('2 мин 6 сек')).toBeInTheDocument();
expect(screen.getByText('2 мин 48 сек')).toBeInTheDocument();
});
Another test is to make sure that if a user clicks on a day, the date of the day becomes active:
it('dispatches setCurrentDay action when a day is clicked', () => {
render(
<Provider store={store}>
<WeekChart />
</Provider>,
);
fireEvent.click(screen.getByText('Пн'));
const actions = store.getActions();
expect(actions).toEqual([setCurrentDay(initialState.stats[1])]);
});
In this snippet, mock store validates a synchronous action and the payload of the expected action. More details on the method is here.
Coverage
Jest is configured to run with cobertura
coverage:
/** @type {import('ts-jest').JestConfigWithTsJest} */
// eslint-disable-next-line
module.exports = {
//...
collectCoverage: true,
collectCoverageFrom: ['./src/**'],
coverageReporters: ['text', 'cobertura'],
coveragePathIgnorePatterns: [
'index.js',
'index.jsx',
'index.ts',
'/node_modules/',
],
};