Mika A.

Tester un changement d'état dans un composant avec enzyme

April 24, 2018

En travaillant actuellement sur un projet React, je suis tombé sur une problématique dans un de mes tests fonctionnels où nous utilisons enzyme. Je vous explique tout cela.

Un composant pour démontrer le problème

Imaginez un composant React qui affiche un texte particulier quand l’état interne du composant a une certaine valeur. Voici un exemple trivial: nous voulons afficher le texte ’😭 It\‘s too late to apologize.’ quand l’état expired du composant est à true. La contrainte est que nous mettons cette valeur à true 1h après avoir monté le composant.

import React from 'react'

class Timer extends React.Component{
  constructor(props) {
    super(props);
    this.state = {
      expired: false,
    }
  }

  componentDidMount() {
    setTimeout(() => {
      this.setState({ expired: true })
    }, 3600 * 1000);
  }

  render() {
    return (
      <div>
        {this.state.expired ? '😭 It's too late to apologize.' : '😇 Everything is fine.'}
      </div>
    );
  }
}

export default Timer;

Imaginez maintenant que je veuille mettre en place un test qui m’assure que ce fonctionnement est toujours valide. En théorie, il faudrait monter le composant et attendre 1h avant de voir si le texte est correctement mis à jour. Il y a surement une manière plus rapide pour gérer cela et c’est là que enzyme montre toute sa force.

Enzyme et setState à la rescousse

Nous allons mettre en place un test avec enzyme et simuler le changement d’état dans le composant avant de vérifier le texte qui s’affiche. Pour vérifier le texte par défaut, c’est assez simple:

import React from 'react';
import { shallow, configure } from 'enzyme';
import Adapter from 'enzyme-adapter-react-16';
import Timer from './Timer';

configure({ adapter: new Adapter() });

it('should display nice text by default', () => {
  const timer = shallow(<Timer />);
  expect(timer.text()).toBe('😇 Everything is fine.');
});

Maintenant, pour pouvoir simuler le changement d’état (expired) sans attendre 1h, nous allons utiliser setState de enzyme et vérifier ensuite que le texte affiché est bien le second.

it('should display bad text when expired', () => {
  const timer = shallow(<Timer />);
  timer.setState({ expired: true });
  expect(timer.text()).toBe('😭 It's too late to apologize.');
});

Maintenant, nos tests valident bien le changement d’état dans le composant sans trop de difficulté. N’hésitez pas à laisser un commentaire pour en discuter.


Written by @mikaoelitiana who lives and works in Paris, building useful things with React, Typescript and more.Follow me on Twitter or Github