RSpec - Crochets

Lorsque vous écrivez des tests unitaires, il est souvent pratique d'exécuter le code de configuration et de démontage avant et après vos tests. Le code de configuration est le code qui configure ou «définit» les conditions d'un test. Le code de démontage effectue le nettoyage, il s'assure que l'environnement est dans un état cohérent pour les tests suivants.

De manière générale, vos tests doivent être indépendants les uns des autres. Lorsque vous exécutez une suite complète de tests et que l'un d'entre eux échoue, vous voulez être sûr qu'il a échoué parce que le code qu'il teste a un bogue, pas parce que le test précédent a laissé l'environnement dans un état incohérent.

Les hooks les plus couramment utilisés dans RSpec sont les hooks avant et après. Ils fournissent un moyen de définir et d'exécuter le code d'installation et de démontage dont nous avons parlé ci-dessus. Considérons cet exemple de code -

class SimpleClass 
   attr_accessor :message 
   
   def initialize() 
      puts "\nCreating a new instance of the SimpleClass class" 
      @message = 'howdy' 
   end 
   
   def update_message(new_message) 
      @message = new_message 
   end 
end 

describe SimpleClass do 
   before(:each) do 
      @simple_class = SimpleClass.new 
   end 
   
   it 'should have an initial message' do 
      expect(@simple_class).to_not be_nil
      @simple_class.message = 'Something else. . .' 
   end 
   
   it 'should be able to change its message' do
      @simple_class.update_message('a new message')
      expect(@simple_class.message).to_not be 'howdy' 
   end
end

Lorsque vous exécutez ce code, vous obtiendrez la sortie suivante -

Creating a new instance of the SimpleClass class 
. 
Creating a new instance of the SimpleClass class 
. 
Finished in 0.003 seconds (files took 0.11401 seconds to load) 
2 examples, 0 failures

Regardons de plus près ce qui se passe. La méthode before (: each) est l'endroit où nous définissons le code de configuration. Lorsque vous passez l'argument: each, vous indiquez à la méthode before de s'exécuter avant chaque exemple de votre groupe d'exemples, c'est-à-dire les deux qu'elle bloque à l'intérieur du bloc describe dans le code ci-dessus.

Dans la ligne: @simple_class = SimpleClass.new, nous créons une nouvelle instance de la classe SimpleClass et l'affectons à une variable d'instance d'un objet. Quel objet vous demandez-vous? RSpec crée une classe spéciale dans les coulisses dans la portée du bloc describe. Cela vous permet d'affecter des valeurs aux variables d'instance de cette classe, auxquelles vous pouvez accéder dans les blocs informatiques de vos exemples. Cela facilite également l'écriture de code plus propre dans nos tests. Si chaque test (exemple) nécessite une instance de SimpleClass, nous pouvons placer ce code dans le hook before et ne pas avoir à l'ajouter à chaque exemple.

Notez que, la ligne «Création d'une nouvelle instance de la classe SimpleClass» est écrite deux fois dans la console, cela montre qu'avant l'appel du hook dans chacun des it blocks.

Comme nous l'avons mentionné, RSpec a également un hook après et les hooks avant et après peuvent prendre: tout comme argument. Le hook after s'exécutera après la cible spécifiée. La cible: all signifie que le hook s'exécutera avant / après tous les exemples. Voici un exemple simple qui illustre le moment où chaque hook est appelé.

describe "Before and after hooks" do 
   before(:each) do 
      puts "Runs before each Example" 
   end 
   
   after(:each) do 
      puts "Runs after each Example" 
   end 
   
   before(:all) do 
      puts "Runs before all Examples" 
   end 
   
   after(:all) do 
      puts "Runs after all Examples"
   end 
   
   it 'is the first Example in this spec file' do 
      puts 'Running the first Example' 
   end 
   
   it 'is the second Example in this spec file' do 
      puts 'Running the second Example' 
   end 
end

Lorsque vous exécutez le code ci-dessus, vous verrez cette sortie -

Runs before all Examples 
Runs before each Example 
Running the first Example 
Runs after each Example 
.Runs before each Example 
Running the second Example 
Runs after each Example 
.Runs after all Examples