RSpec - Guide rapide

RSpec est un framework de test unitaire pour le langage de programmation Ruby. RSpec est différent des frameworks xUnit traditionnels tels que JUnit car RSpec est un outil de développement basé sur le comportement. Cela signifie que les tests écrits dans RSpec se concentrent sur le «comportement» d'une application testée. RSpec ne met pas l'accent sur le fonctionnement de l'application, mais plutôt sur la façon dont elle se comporte, en d'autres termes, sur ce que fait réellement l'application.

Environnement RSpec

Tout d'abord, vous devrez installer Ruby sur votre ordinateur. Cependant, si vous ne l'avez pas déjà fait, vous pouvez télécharger et installer Ruby à partir du site Web principal de Ruby - Ruby .

Si vous installez Ruby sur Windows, vous devriez avoir le programme d'installation de Ruby pour Windows ici à - http://www.rubyinstaller.org

Pour ce didacticiel, vous n'aurez besoin que d'un éditeur de texte, tel que le Bloc-notes et une console de ligne de commande. Les exemples ici utiliseront cmd.exe sous Windows.

Pour exécuter cmd.exe, cliquez simplement sur le menu Démarrer et tapez «cmd.exe», puis appuyez sur la touche Retour.

À l'invite de commande dans votre fenêtre cmd.exe, tapez la commande suivante pour voir quelle version de Ruby vous utilisez -

ruby -v

Vous devriez voir la sortie ci-dessous qui ressemble à ceci -

ruby 2.2.3p173 (2015-08-18 revision 51636) [x64-mingw32]

Les exemples de ce tutoriel utiliseront Ruby 2.2.3 mais toute version de Ruby supérieure à 2.0.0 suffira. Ensuite, nous devons installer le gem RSpec pour votre installation Ruby. Un gem est une bibliothèque Ruby que vous pouvez utiliser dans votre propre code. Pour installer une gemme, vous devez utiliser legem commander.

Installons maintenant le gem Rspec. Revenez à votre fenêtre cmd.exe et tapez ce qui suit -

gem install rspec

Vous devriez avoir une liste des gemmes dépendantes qui ont été installées, ce sont des gemmes dont le gem rspec a besoin pour fonctionner correctement. À la fin de la sortie, vous devriez voir quelque chose qui ressemble à ceci -

Done installing documentation for diff-lcs, rspec-support, rspec-mocks,
   rspec-expectations, rspec-core, rspec after 22 seconds 
6 gems installed

Ne vous inquiétez pas, si votre sortie n'est pas exactement la même. De plus, si vous utilisez un ordinateur Mac ou Linux, vous devrez peut-être exécutergem install rspec commande utilisant sudo ou utilisez un outil comme HomeBrew ou RVM pour installer le gem rspec.

Hello World

Pour commencer, créons un répertoire (dossier) pour stocker nos fichiers RSpec. Dans votre fenêtre cmd.exe, tapez ce qui suit -

cd \

Puis tapez -

mkdir rspec_tutorial

Et enfin, tapez -

cd rspec_tutorial

À partir de là, nous allons créer un autre répertoire nommé spec, faites-le en tapant -

mkdir spec

Nous allons stocker nos fichiers RSpec dans ce dossier. Les fichiers RSpec sont appelés «spécifications». Si cela vous semble déroutant, vous pouvez considérer un fichier de spécifications comme un fichier de test. RSpec utilise le terme «spec» qui est une forme abrégée de «spécification».

Comme RSpec est un outil de test BDD, l'objectif est de se concentrer sur ce que fait l'application et si elle suit ou non une spécification. Dans le développement axé sur le comportement, la spécification est souvent décrite en termes de «User Story». RSpec est conçu pour indiquer clairement si le code cible se comporte correctement, en d'autres termes en suivant la spécification.

Revenons à notre code Hello World. Ouvrez un éditeur de texte et ajoutez le code suivant -

class HelloWorld

   def say_hello 
      "Hello World!"
   end
   
end

describe HelloWorld do 
   context “When testing the HelloWorld class” do 
      
      it "should say 'Hello World' when we call the say_hello method" do 
         hw = HelloWorld.new 
         message = hw.say_hello 
         expect(message).to eq "Hello World!"
      end
      
   end
end

Ensuite, enregistrez-le dans un fichier nommé hello_world_spec.rb dans le dossier spec que vous avez créé ci-dessus. Maintenant de retour dans votre fenêtre cmd.exe, exécutez cette commande -

rspec spec spec\hello_world_spec.rb

Lorsque la commande se termine, vous devriez voir une sortie qui ressemble à ceci -

Finished in 0.002 seconds (files took 0.11101 seconds to load) 
1 example, 0 failures

Félicitations, vous venez de créer et d'exécuter votre premier test unitaire RSpec!

Dans la section suivante, nous continuerons à discuter de la syntaxe des fichiers RSpec.

Regardons de plus près le code de notre HelloWorldexemple. Tout d'abord, au cas où ce ne serait pas clair, nous testons la fonctionnalité duHelloWorldclasse. Ceci bien sûr, est une classe très simple qui ne contient qu'une seule méthodesay_hello().

Voici à nouveau le code RSpec -

describe HelloWorld do 
   context “When testing the HelloWorld class” do 
      
      it "The say_hello method should return 'Hello World'" do 
         hw = HelloWorld.new 
         message = hw.say_hello 
         expect(message).to eq "Hello World!" 
      end
      
   end 
end

Le mot-clé décrire

Le mot describeest un mot clé RSpec. Il est utilisé pour définir un «Groupe d'exemples». Vous pouvez considérer un «groupe d'exemples» comme une collection de tests. ledescribeLe mot clé peut prendre un nom de classe et / ou un argument de chaîne. Vous devez également passer un argument de bloc àdescribe, celui-ci contiendra les tests individuels, ou comme ils sont connus dans RSpec, les «exemples». Le bloc est juste un bloc Ruby désigné par le Rubydo/end mots clés.

Le mot-clé de contexte

le context mot-clé est similaire à describe. Il peut également accepter un nom de classe et / ou un argument de chaîne. Vous devez utiliser un bloc aveccontextainsi que. L'idée de contexte est qu'il renferme des tests d'un certain type.

Par exemple, vous pouvez spécifier des groupes d'exemples avec différents contextes comme celui-ci -

context “When passing bad parameters to the foobar() method” 
context “When passing valid parameters to the foobar() method” 
context “When testing corner cases with the foobar() method”

le context Le mot-clé n'est pas obligatoire, mais il permet d'ajouter plus de détails sur les exemples qu'il contient.

Le mot-clé it

Le mot itest un autre mot clé RSpec utilisé pour définir un «exemple». Un exemple est essentiellement un test ou un cas de test. Encore une fois, commedescribe et context, it accepte à la fois le nom de classe et les arguments de chaîne et doit être utilisé avec un argument de bloc, désigné par do/end. Dans le cas deit, il est habituel de ne transmettre qu'une chaîne et un argument de bloc. L'argument chaîne utilise souvent le mot «devrait» et est destiné à décrire le comportement spécifique qui devrait se produire dans leit block. En d'autres termes, il décrit que le résultat attendu est pour l'exemple.

Noter la it block de notre exemple HelloWorld -

it "The say_hello method should return 'Hello World'" do

La chaîne indique clairement ce qui doit se passer lorsque nous appelons say hello sur une instance de la classe HelloWorld. Cette partie de la philosophie RSpec, un exemple n'est pas seulement un test, c'est aussi une spécification (une spécification). En d'autres termes, un exemple documente et teste à la fois le comportement attendu de votre code Ruby.

Le mot-clé attendu

le expectLe mot-clé est utilisé pour définir une «attente» dans RSpec. Il s'agit d'une étape de vérification où nous vérifions qu'une condition attendue spécifique a été remplie.

De notre exemple HelloWorld, nous avons -

expect(message).to eql "Hello World!"

L'idée avec expectdéclarations, c'est qu'ils se lisent comme un anglais normal. Vous pouvez dire cela à voix haute comme «Attendez-vous à ce que le message variable soit égal à la chaîne« Hello World »». L'idée est que c'est descriptif et aussi facile à lire, même pour les parties prenantes non techniques telles que les chefs de projet.

The to keyword

le to Le mot-clé est utilisé dans le cadre de expectdéclarations. Notez que vous pouvez également utiliser lenot_tomot-clé pour exprimer le contraire, lorsque vous voulez que l'attente soit fausse. Vous pouvez voir que to est utilisé avec un point,expect(message).to,car il s'agit en fait d'une méthode Ruby classique. En fait, tous les mots-clés RSpec ne sont en réalité que des méthodes Ruby.

The eql keyword

le eqlmot-clé est un mot-clé RSpec spécial appelé Matcher. Vous utilisez Matchers pour spécifier le type de condition que vous testez pour être vrai (ou faux).

Dans notre HelloWorld expect déclaration, il est clair que eqlsignifie l'égalité des chaînes. Notez qu'il existe différents types d'opérateurs d'égalité dans Ruby et par conséquent différents Matchers correspondants dans RSpec. Nous explorerons les différents types de Matchers dans une section ultérieure.

Dans ce chapitre, nous allons créer une nouvelle classe Ruby, l'enregistrer dans son propre fichier et créer un fichier de spécification séparé pour tester cette classe.

Premièrement, dans notre nouvelle classe, il s'appelle StringAnalyzer. C'est une classe simple qui, vous l'avez deviné, analyse les chaînes. Notre classe n'a qu'une seule méthodehas_vowels?qui, comme son nom l'indique, renvoie true si une chaîne contient des voyelles et false si ce n'est pas le cas. Voici l'implémentation pourStringAnalyzer -

class StringAnalyzer 
   def has_vowels?(str) 
      !!(str =~ /[aeio]+/i) 
   end 
end

Si vous avez suivi la section HelloWorld, vous avez créé un dossier appelé C: \ rspec_tutorial \ spec.

Supprimez le fichier hello_world.rb si vous l'avez et enregistrez le code StringAnalyzer ci-dessus dans un fichier appelé string_analyzer.rb dans le dossier C: \ rspec_tutorial \ spec.

Voici la source de notre fichier de spécifications pour tester StringAnalyzer -

require 'string_analyzer' 

describe StringAnalyzer do 
   context "With valid input" do 
      
      it "should detect when a string contains vowels" do 
         sa = StringAnalyzer.new 
         test_string = 'uuu' 
         expect(sa.has_vowels? test_string).to be true 
      end 
		
      it "should detect when a string doesn't contain vowels" do 
         sa = StringAnalyzer.new 
         test_string = 'bcdfg' 
         expect(sa.has_vowels? test_string).to be false
      end 
      
   end 
end

Enregistrez-le dans le même répertoire de spécification, en lui donnant le nom string_analyzer_test.rb.

Dans votre fenêtre cmd.exe, accédez au dossier C: \ rspec_tutorial et exécutez cette commande: dir spec

Vous devriez voir ce qui suit -

Répertoire de C: \ rspec_tutorial \ spec

09/13/2015 08:22 AM  <DIR>    .
09/13/2015 08:22 AM  <DIR>    ..
09/12/2015 11:44 PM                 81 string_analyzer.rb
09/12/2015 11:46 PM              451 string_analyzer_test.rb

Maintenant, nous allons exécuter nos tests, exécutez cette commande: rspec spec

Lorsque vous transmettez le nom d'un dossier à rspec, il exécute tous les fichiers de spécifications à l'intérieur du dossier. Vous devriez voir ce résultat -

No examples found.

Finished in 0 seconds (files took 0.068 seconds to load)
0 examples, 0 failures

La raison pour laquelle cela s'est produit est que, par défaut, rspecexécute uniquement les fichiers dont les noms se terminent par «_spec.rb». Renommez string_analyzer_test.rb en string_analyzer_spec.rb. Vous pouvez le faire facilement en exécutant cette commande -

ren spec\string_analyzer_test.rb string_analyzer_spec.rb

Maintenant, cours rspec spec encore une fois, vous devriez voir une sortie qui ressemble à ceci -

F.
Failures:

   1) StringAnalyzer With valid input should detect when a string contains vowels
      Failure/Error: expect(sa.has_vowels? test_string).to be true 
         expected true
            got false
      # ./spec/string_analyzer_spec.rb:9:in `block (3 levels) in <top (required)>'

Finished in 0.015 seconds (files took 0.12201 seconds to load)
2 examples, 1 failure

Failed examples:
rspec ./spec/string_analyzer_spec.rb:6 # StringAnalyzer With valid 
   input should detect when a string contains vowels
Do you see what just happened? Our spec failed because we have a bug in 
   StringAnalyzer. The bug is simple to fix, open up string_analyzer.rb
   in a text editor and change this line:
!!(str =~ /[aeio]+/i)
to this:
!!(str =~ /[aeiou]+/i)

Maintenant, enregistrez les modifications que vous venez de faire dans string_analyizer.rb et exécutez à nouveau la commande rspec spec, vous devriez maintenant voir une sortie qui ressemble à -

..
Finished in 0.002 seconds (files took 0.11401 seconds to load)
2 examples, 0 failures

Félicitations, les exemples (tests) de votre fichier de spécifications sont maintenant réussis. Nous avons corrigé un bug dans l'expression régulière qui a la méthode des voyelles mais nos tests sont loin d'être terminés.

Il serait logique d'ajouter d'autres exemples qui testent différents types de chaînes d'entrée avec la méthode has voyelles.

Le tableau suivant montre certaines des permutations qui pourraient être ajoutées dans de nouveaux exemples (il bloque)

Chaîne d'entrée La description Résultat attendu avec has_vowels?
'aaa', 'eee', 'iii', 'o' Une seule voyelle et pas d'autres lettres. vrai
'abcefg' 'Au moins une voyelle et quelques consonnes' vrai
«mnklp» Seulement des consonnes. faux
'' Chaîne vide (pas de lettres) faux
'abcde55345 & ??' Voyelles, consonnes, chiffres et caractères de ponctuation. vrai
«423432 %%% ^ &» Chiffres et caractères de ponctuation uniquement. faux
«AEIOU» Voyelles majuscules uniquement. vrai
«AeiOuuuA» Voyelles majuscules et minuscules uniquement. vrai
«AbCdEfghI» Voyelles et consonnes majuscules et minuscules. vrai
«BCDFG» Consonnes majuscules uniquement. faux
'' Caractères d'espaces blancs uniquement. faux

C'est à vous de décider quels exemples ajouter à votre fichier de spécifications. Il existe de nombreuses conditions à tester, vous devez déterminer quel sous-ensemble de conditions est le plus important et tester votre code le mieux.

le rspec La commande offre de nombreuses options différentes, pour les voir toutes, tapez rspec-Aidez-moi. Le tableau suivant répertorie les options les plus courantes et décrit leur fonction.

Sr.No. Option / drapeau et description
1

-I PATH

Ajoute PATH au chemin de chargement (requis) qui rspec utilise lors de la recherche de fichiers source Ruby.

2

-r, --require PATH

Ajoute un fichier source spécifique à exiger dans vos spécifications. des dossiers).

3

--fail-fast

Avec cette option, rspec arrêtera d'exécuter des specs après l'échec du premier exemple. Par défaut, rspec exécute tous les fichiers de spécifications spécifiés, quel que soit le nombre d'échecs.

4

-f, --format FORMATTER

Cette option vous permet de spécifier différents formats de sortie. Consultez la section sur les formateurs pour plus de détails sur les formats de sortie.

5

-o, --out FILE

Cette option demande à rspec d'écrire les résultats du test dans le fichier de sortie FILE plutôt que dans la sortie standard.

6

-c, --color

Active la couleur dans la sortie de rspec. Les résultats des exemples réussis s'afficheront en texte vert, les échecs seront imprimés en texte rouge.

sept

-b, --backtrace

Affiche les retours arrière d'erreur complets dans la sortie de rspec.

8

-w, --warnings

Affiche les avertissements Ruby dans la sortie de rspec.

9

-P, --pattern PATTERN

Chargez et exécutez les fichiers de spécifications qui correspondent au modèle PATTERN. Par exemple, si vous passez -p «* .rb», rspec exécutera tous les fichiers Ruby, pas seulement ceux qui se terminent par «_spec.rb».

dix

-e, --example STRING

Cette option demande à rspec d'exécuter tous les exemples qui contiennent le texte STRING dans leurs descriptions.

11

-t, --tag TAG

Avec cette option, rspec n'exécutera que les exemples contenant la balise TAG. Notez que TAG est spécifié comme un symbole Ruby. Voir la section sur les balises RSpec pour plus de détails.

Si vous vous souvenez de notre exemple Hello World original, il contenait une ligne qui ressemblait à ceci -

expect(message).to eq "Hello World!"

Le mot-clé eql est un RSpec«Matcher». Ici, nous allons présenter les autres types de matchers dans RSpec.

Matchers égalité / identité

Matchers pour tester l'égalité d'objet ou de valeur.

Matcher La description Exemple
eq Passe lorsque réel == attendu expect (réel) .to eq attendu
eql Passe quand actual.eql? (Attendu) expect (réel) .to eql attendu
être Passe quand actual.equal? ​​(Attendu) s'attendre (réel) .à prévoir
égal Passe également quand actual.equal? ​​(Attendu) attendre (réel). égaler attendu

Exemple

describe "An example of the equality Matchers" do 

   it "should show how the equality Matchers work" do 
      a = "test string" 
      b = a 
      
      # The following Expectations will all pass 
      expect(a).to eq "test string" 
      expect(a).to eql "test string" 
      expect(a).to be b 
      expect(a).to equal b 
   end
   
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le nombre de secondes peut être légèrement différent sur votre ordinateur -

.
Finished in 0.036 seconds (files took 0.11901 seconds to load)
1 example, 0 failures

Matchers de comparaison

Matchers pour comparer aux valeurs.

Matcher La description Exemple
> Passe quand réel> attendu s'attendre (réel). être> attendu
> = Passe lorsque réel> = attendu s'attendre (réel). être> = attendu
< Passe lorsque réel <attendu s'attendre (réel). être <attendu
<= Passe lorsque réel <= attendu s'attendre (réel). être <= attendu
be_between inclusif Passe lorsque le réel est <= min et> = max s'attendre à (réel) .être_entre (min, max) .inclusif
be_between exclusif Passe lorsque le réel est <min et> max s'attendre à (réel) .être_entre (min, max) .exclusif
rencontre Passe lorsque le réel correspond à une expression régulière attendre (réel) .pour correspondre (/ regex /)

Exemple

describe "An example of the comparison Matchers" do

   it "should show how the comparison Matchers work" do
      a = 1
      b = 2
      c = 3		
      d = 'test string'
      
      # The following Expectations will all pass
      expect(b).to be > a
      expect(a).to be >= a 
      expect(a).to be < b 
      expect(b).to be <= b 
      expect(c).to be_between(1,3).inclusive 
      expect(b).to be_between(1,3).exclusive 
      expect(d).to match /TEST/i 
   end
   
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le nombre de secondes peut être légèrement différent sur votre ordinateur -

. 
Finished in 0.013 seconds (files took 0.11801 seconds to load) 
1 example, 0 failures

Matchers de classe / type

Matchers pour tester le type ou la classe d'objets.

Matcher La description Exemple
be_instance_of Passe lorsque actual est une instance de la classe attendue. expect (réel) .to be_instance_of (attendu)
be_kind_of Passe lorsque actual est une instance de la classe attendue ou de l'une de ses classes parentes. expect (réel) .to be_kind_of (attendu)
répondre à Passe lorsque le réel répond à la méthode spécifiée. expect (réel) .to respond_to (attendu)

Exemple

describe "An example of the type/class Matchers" do
 
   it "should show how the type/class Matchers work" do
      x = 1 
      y = 3.14 
      z = 'test string' 
      
      # The following Expectations will all pass
      expect(x).to be_instance_of Fixnum 
      expect(y).to be_kind_of Numeric 
      expect(z).to respond_to(:length) 
   end
   
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le nombre de secondes peut être légèrement différent sur votre ordinateur -

. 
Finished in 0.002 seconds (files took 0.12201 seconds to load) 
1 example, 0 failures

Correspondants vrai / faux / nul

Matchers pour tester si une valeur est vraie, fausse ou nulle.

Matcher La description Exemple
Sois sincère Passe lorsque réel == vrai s'attendre (réel) .à être vrai
être faux Passe lorsque réel == false s'attendre à (réel) .être faux
be_truthy Passe lorsque le réel n'est pas faux ou nul s'attendre à (réel) .être_vérité
be_falsey Passe lorsque le réel est faux ou nul s'attendre à (réel) .être_falsey
be_nil Passe lorsque le réel est nul attendre (réel) .être_aucun

Exemple

describe "An example of the true/false/nil Matchers" do
   it "should show how the true/false/nil Matchers work" do
      x = true 
      y = false 
      z = nil 
      a = "test string" 
      
      # The following Expectations will all pass
      expect(x).to be true 
      expect(y).to be false 
      expect(a).to be_truthy 
      expect(z).to be_falsey 
      expect(z).to be_nil 
   end 
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le nombre de secondes peut être légèrement différent sur votre ordinateur -

. 
Finished in 0.003 seconds (files took 0.12301 seconds to load) 
1 example, 0 failures

Correspondants d'erreur

Matchers pour les tests, lorsqu'un bloc de code déclenche une erreur.

Matcher La description Exemple
rise_error (ErrorClass) Passe lorsque le bloc génère une erreur de type ErrorClass. attendre {block} .to rise_error (ErrorClass)
rise_error ("message d'erreur") Passe lorsque le bloc soulève une erreur avec le message «message d'erreur». attendre {block} .to rise_error ("message d'erreur")
rise_error (ErrorClass, "message d'erreur") Passe lorsque le bloc déclenche une erreur de type ErrorClass avec le message «message d'erreur» attendez {block} .to rise_error (ErrorClass, "message d'erreur")

Exemple

Enregistrez le code suivant dans un fichier avec le nom error_matcher_spec.rb et exécutez-le avec cette commande - rspec error_matcher_spec.rb.

describe "An example of the error Matchers" do 
   it "should show how the error Matchers work" do 
      
      # The following Expectations will all pass 
      expect { 1/0 }.to raise_error(ZeroDivisionError)
      expect { 1/0 }.to raise_error("divided by 0") 
      expect { 1/0 }.to raise_error("divided by 0", ZeroDivisionError) 
   end 
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le nombre de secondes peut être légèrement différent sur votre ordinateur -

. 
Finished in 0.002 seconds (files took 0.12101 seconds to load) 
1 example, 0 failures

Dans ce chapitre, nous aborderons RSpec Doubles, également connu sous le nom de RSpec Mocks. Un Double est un objet qui peut «remplacer» un autre objet. Vous vous demandez probablement ce que cela signifie exactement et pourquoi vous en auriez besoin.

Disons que vous créez une application pour une école et que vous avez une classe représentant une classe d'élèves et une autre classe pour les élèves, c'est-à-dire que vous avez une classe Classroom et une classe Student. Vous devez d'abord écrire le code de l'une des classes, alors disons que, commencez par la classe Classroom -

class ClassRoom 
   def initialize(students) 
      @students = students 
   end 
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
end

Il s'agit d'une classe simple, elle a une méthode list_student_names, qui renvoie une chaîne de noms d'étudiants délimitée par des virgules. Maintenant, nous voulons créer des tests pour cette classe, mais comment faire cela si nous n'avons pas encore créé la classe Student? Nous avons besoin d'un test Double.

De plus, si nous avons une classe «factice» qui se comporte comme un objet Student, nos tests ClassRoom ne dépendront pas de la classe Student. Nous appelons cet isolement de test.

Si nos tests ClassRoom ne reposent sur aucune autre classe, alors lorsqu'un test échoue, nous pouvons savoir immédiatement qu'il y a un bogue dans notre classe ClassRoom et pas dans une autre classe. Gardez à l'esprit que, dans le monde réel, vous construisez peut-être une classe qui a besoin d'interagir avec une autre classe écrite par quelqu'un d'autre.

C'est là que RSpec Doubles (mocks) devient utile. Notre méthode list_student_names appelle la méthode name sur chaque objet Student dans sa variable membre @students. Par conséquent, nous avons besoin d'un Double qui implémente une méthode de nom.

Voici le code pour ClassRoom avec un exemple RSpec (test), mais notez qu'il n'y a pas de classe Student définie -

class ClassRoom 
   def initialize(students) 
      @students = students 
   end
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
end

describe ClassRoom do 
   it 'the list_student_names method should work correctly' do 
      student1 = double('student') 
      student2 = double('student') 
      
      allow(student1).to receive(:name) { 'John Smith'} 
      allow(student2).to receive(:name) { 'Jill Smith'} 
      
      cr = ClassRoom.new [student1,student2]
      expect(cr.list_student_names).to eq('John Smith,Jill Smith') 
   end 
end

Lorsque le code ci-dessus est exécuté, il produira la sortie suivante. Le temps écoulé peut être légèrement différent sur votre ordinateur -

. 
Finished in 0.01 seconds (files took 0.11201 seconds to load) 
1 example, 0 failures

Comme vous pouvez le voir, en utilisant un test doublevous permet de tester votre code même s'il s'appuie sur une classe indéfinie ou indisponible. En outre, cela signifie qu'en cas d'échec du test, vous pouvez dire tout de suite que c'est à cause d'un problème dans votre classe et non d'une classe écrite par quelqu'un d'autre.

Si vous avez déjà lu la section sur RSpec Doubles (aka Mocks), alors vous avez déjà vu RSpec Stubs. Dans RSpec, un stub est souvent appelé un stub de méthode, c'est un type spécial de méthode qui «remplace» une méthode existante, ou une méthode qui n'existe même pas encore.

Voici le code de la section sur RSpec Doubles -

class ClassRoom 
   def initialize(students) 
      @students = students 
   End
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
end 

describe ClassRoom do 
   it 'the list_student_names method should work correctly' do 
      student1 = double('student') 
      student2 = double('student') 
      
      allow(student1).to receive(:name) { 'John Smith'}
      allow(student2).to receive(:name) { 'Jill Smith'} 
      
      cr = ClassRoom.new [student1,student2]
      expect(cr.list_student_names).to eq('John Smith,Jill Smith') 
   end 
end

Dans notre exemple, la méthode allow () fournit les stubs de méthode dont nous avons besoin pour tester la classe ClassRoom. Dans ce cas, nous avons besoin d'un objet qui agira comme une instance de la classe Student, mais cette classe n'existe pas (encore). Nous savons que la classe Student doit fournir une méthode name () et nous utilisons allow () pour créer un stub de méthode pour name ().

Une chose à noter est que la syntaxe de RSpec a un peu changé au fil des ans. Dans les anciennes versions de RSpec, les stubs de méthode ci-dessus seraient définis comme ceci -

student1.stub(:name).and_return('John Smith') 
student2.stub(:name).and_return('Jill Smith')

Prenons le code ci-dessus et remplaçons les deux allow() lignes avec l'ancienne syntaxe RSpec -

class ClassRoom 
   def initialize(students) 
      @students = students 
   end 
   
   def list_student_names 
      @students.map(&:name).join(',') 
   end 
	
end 

describe ClassRoom do 
   it 'the list_student_names method should work correctly' do 
      student1 = double('student') 
      student2 = double('student')
      
      student1.stub(:name).and_return('John Smith')
      student2.stub(:name).and_return('Jill Smith') 
      
      cr = ClassRoom.new [student1,student2] 
      expect(cr.list_student_names).to eq('John Smith,Jill Smith') 
   end 
end

Vous verrez cette sortie lorsque vous exécuterez le code ci-dessus -

.
Deprecation Warnings:

Using `stub` from rspec-mocks' old `:should` syntax without explicitly 
   enabling the syntax is deprec 

ated. Use the new `:expect` syntax or explicitly enable `:should` instead. 
   Called from C:/rspec_tuto 

rial/spec/double_spec.rb:15:in `block (2 levels) in <top (required)>'.
If you need more of the backtrace for any of these deprecations 
   to identify where to make the necessary changes, you can configure 

`config.raise_errors_for_deprecations!`, and it will turn the 
   deprecation warnings into errors, giving you the full backtrace.

1 deprecation warning total

Finished in 0.002 seconds (files took 0.11401 seconds to load)
1 example, 0 failures

Il est recommandé d'utiliser la nouvelle syntaxe allow () lorsque vous avez besoin de créer des stubs de méthode dans vos exemples RSpec, mais nous avons fourni l'ancien style ici afin que vous le reconnaissiez si vous le voyez.

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'assignons à 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 de description. Cela vous permet d'attribuer 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éer 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

Les balises RSpec offrent un moyen simple d'exécuter des tests spécifiques dans vos fichiers de spécifications. Par défaut, RSpec exécutera tous les tests dans les fichiers de spécification qu'il exécute, mais vous n'aurez peut-être besoin d'en exécuter qu'un sous-ensemble. Disons que vous avez des tests qui s'exécutent très rapidement et que vous venez de modifier le code de votre application et que vous souhaitez simplement exécuter les tests rapides, ce code vous montrera comment faire cela avec les balises RSpec.

describe "How to run specific Examples with Tags" do 
   it 'is a slow test', :slow = > true do 
      sleep 10 
      puts 'This test is slow!' 
   end 
   
   it 'is a fast test', :fast = > true do 
      puts 'This test is fast!' 
   end 
end

Maintenant, enregistrez le code ci-dessus dans un nouveau fichier appelé tag_spec.rb. Depuis la ligne de commande, exécutez cette commande: rspec --tag slow tag_spec.rb

Vous verrez cette sortie -

Options d'exécution: include {: slow => true}

This test is slow! 
. 
Finished in 10 seconds (files took 0.11601 seconds to load) 
1 example, 0 failures

Ensuite, exécutez cette commande: rspec --tag fast tag_spec.rb

Vous verrez cette sortie -

Run options: include {:fast = >true} 
This test is fast! 
. 
Finished in 0.001 seconds (files took 0.11201 seconds to load) 
1 example, 0 failures

Comme vous pouvez le voir, les balises RSpec rendent très facile un sous-ensemble de tests!

L'une des forces de RSpec est qu'il fournit de nombreuses façons d'écrire des tests, des tests propres. Lorsque vos tests sont courts et épurés, il devient plus facile de se concentrer sur le comportement attendu et non sur les détails de la rédaction des tests. Les sujets RSpec sont encore un autre raccourci vous permettant d'écrire des tests simples et simples.

Considérez ce code -

class Person 
   attr_reader :first_name, :last_name 
   
   def initialize(first_name, last_name) 
      @first_name = first_name 
      @last_name = last_name 
   end 
end 

describe Person do 
   it 'create a new person with a first and last name' do
      person = Person.new 'John', 'Smith'
      
      expect(person).to have_attributes(first_name: 'John') 
      expect(person).to have_attributes(last_name: 'Smith') 
   end 
end

C'est en fait assez clair en l'état, mais nous pourrions utiliser la fonction de sujet de RSpec pour réduire la quantité de code dans l'exemple. Nous faisons cela en déplaçant l'instanciation de l'objet personne dans la ligne de description.

class Person 
   attr_reader :first_name, :last_name 
   
   def initialize(first_name, last_name) 
      @first_name = first_name 
      @last_name = last_name 
   end 
	
end 

describe Person.new 'John', 'Smith' do 
   it { is_expected.to have_attributes(first_name: 'John') } 
   it { is_expected.to have_attributes(last_name: 'Smith') }
end

Lorsque vous exécutez ce code, vous verrez cette sortie -

.. 
Finished in 0.003 seconds (files took 0.11201 seconds to load) 
2 examples, 0 failures

Notez à quel point le deuxième exemple de code est plus simple. Nous avons pris celuiit block dans le premier exemple et l'a remplacé par deux it blocks qui finissent par nécessiter moins de code et sont tout aussi clairs.

Parfois, vos exemples RSpec nécessitent un moyen simple de partager du code réutilisable. La meilleure façon d'y parvenir est d'utiliser les Helpers. Les Helpers sont essentiellement des méthodes Ruby régulières que vous partagez à travers des exemples. Pour illustrer les avantages de l'utilisation des helpers, considérons ce code -

class Dog 
   attr_reader :good_dog, :has_been_walked 
   
   def initialize(good_or_not) 
      @good_dog = good_or_not 
      @has_been_walked = false 
   end 
   
   def walk_dog 
      @has_been_walked = true 
   end 
end 

describe Dog do 
   it 'should be able to create and walk a good dog' do 
      dog = Dog.new(true) 
      dog.walk_dog 
      
      expect(dog.good_dog).to be true
      expect(dog.has_been_walked).to be true 
   end 
   
   it 'should be able to create and walk a bad dog' do 
      dog = Dog.new(false) 
      dog.walk_dog 

      expect(dog.good_dog).to be false
      expect(dog.has_been_walked).to be true 
 
   end 
end

Ce code est clair, mais c'est toujours une bonne idée de réduire autant que possible le code répété. Nous pouvons prendre le code ci-dessus et réduire une partie de cette répétition avec une méthode d'assistance appelée create_and_walk_dog ().

class Dog
   attr_reader :good_dog, :has_been_walked 
   
   def initialize(good_or_not)
      @good_dog = good_or_not 
      @has_been_walked = false 
   end 
   
   def walk_dog 
      @has_been_walked = true 
   end 
end 

describe Dog do 
   def create_and_walk_dog(good_or_bad)
      dog = Dog.new(good_or_bad)
      dog.walk_dog
      return dog 
   end 
   
   it 'should be able to create and walk a good dog' do
      dog = create_and_walk_dog(true)
      
      expect(dog.good_dog).to be true
      expect(dog.has_been_walked).to be true 
   end 
   
   it 'should be able to create and walk a bad dog' do 
      dog = create_and_walk_dog(false)
      
      expect(dog.good_dog).to be false
      expect(dog.has_been_walked).to be true 
   end 
end

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

.. 
Finished in 0.002 seconds (files took 0.11401 seconds to load) 
2 examples, 0 failures

Comme vous pouvez le voir, nous avons pu pousser la logique de création et de marche d'un objet chien dans un Helper, ce qui permet à nos exemples d'être plus courts et plus propres.

RSpec est un outil flexible et puissant. La fonctionnalité de métadonnées de RSpec ne fait pas exception. Les métadonnées font généralement référence aux «données sur les données». Dans RSpec, cela signifie des données sur votredescribe, context et it blocks.

Jetons un coup d'œil à un exemple -

RSpec.describe "An Example Group with a metadata variable", :foo => 17 do 
   context 'and a context with another variable', :bar => 12 do 
      
      it 'can access the metadata variable of the outer Example Group' do |example| 
         expect(example.metadata[:foo]).to eq(17) 
      end
      
      it 'can access the metadata variable in the context block' do |example|  
         expect(example.metadata[:bar]).to eq(12) 
      end 
      
   end 
end

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

.. 
Finished in 0.002 seconds (files took 0.11301 seconds to load) 
2 examples, 0 failures

Les métadonnées permettent d'attribuer des variables à différentes portées dans vos fichiers RSpec. La variable example.metadata est un hachage Ruby qui contient d'autres informations sur vos groupes d'exemples et d'exemples.

Par exemple, réécrivons le code ci-dessus pour ressembler à ceci -

RSpec.describe "An Example Group with a metadata variable", :foo => 17 do
   context 'and a context with another variable', :bar => 12 do 
      
      it 'can access the metadata variable in the context block' do |example|
         expect(example.metadata[:foo]).to eq(17) 
         expect(example.metadata[:bar]).to eq(12) 
         example.metadata.each do |k,v|
         puts "#{k}: #{v}"
      end
		
   end 
end

Lorsque nous exécutons ce code, nous voyons toutes les valeurs dans le hachage example.metadata -

.execution_result: #<RSpec::Core::Example::ExecutionResult:0x00000002befd50>
block: #<Proc:0x00000002bf81a8@C:/rspec_tutorial/spec/metadata_spec.rb:7>
description_args: ["can access the metadata variable in the context block"]
description: can access the metadata variable in the context block
full_description: An Example Group with a metadata variable and a context 
   with another variable can access the metadata variable in the context block
described_class:
file_path: ./metadata_spec.rb
line_number: 7
location: ./metadata_spec.rb:7
absolute_file_path: C:/rspec_tutorial/spec/metadata_spec.rb
rerun_file_path: ./metadata_spec.rb
scoped_id: 1:1:2
foo: 17
bar: 12
example_group:
{:execution_result=>#<RSpec::Core::Example::ExecutionResult:
   0x00000002bfa0e8>, :block=>#<
   Proc:0x00000002bfac00@C:/rspec_tutorial/spec/metadata_spec.rb:2>, 
   :description_args=>["and a context with another variable"], 
	
   :description=>"and a context with another variable", 
   :full_description=>"An Example Group with a metadata variable
   and a context with another variable", :described_class=>nil, 
      :file_path=>"./metadata_spec.rb", 
		
   :line_number=>2, :location=>"./metadata_spec.rb:2", 
      :absolute_file_path=>"C:/rspec_tutorial/spec/metadata_spec.rb",
      :rerun_file_path=>"./metadata_spec.rb", 
		
   :scoped_id=>"1:1", :foo=>17, :parent_example_group=>
      {:execution_result=>#<
      RSpec::Core::Example::ExecutionResult:0x00000002c1f690>, 
      :block=>#<Proc:0x00000002baff70@C:/rspec_tutorial/spec/metadata_spec.rb:1>
      , :description_args=>["An Example Group with a metadata variable"], 
		
   :description=>"An Example Group with a metadata variable", 
   :full_description=>"An Example Group with a metadata variable", 
	:described_class=>nil, :file_path=>"./metadata_spec.rb", 
   :line_number=>1, :location=>"./metadata_spec.rb:1",
   :absolute_file_path=>
	
   "C:/rspec_tutorial/spec/metadata_spec.rb", 
   :rerun_file_path=>"./metadata_spec.rb", 
   :scoped_id=>"1", :foo=>17}, 
   :bar=>12}shared_group_inclusion_backtrace: [] 
	
last_run_status: unknown .
.
Finished in 0.004 seconds (files took 0.11101 seconds to load) 
2 examples, 0 failures

Très probablement, vous n'aurez pas besoin d'utiliser toutes ces métadonnées, mais regardez la valeur de la description complète -

Un groupe d'exemples avec une variable de métadonnées et un contexte avec une autre variable peut accéder à la variable de métadonnées dans le bloc de contexte.

Il s'agit d'une phrase créée à partir de la description du bloc de description + de sa description de bloc de contexte contenue + de la description du it block.

Ce qui est intéressant à noter ici, c'est que ces trois chaînes se lisent ensemble comme une phrase anglaise normale. . . qui est l'une des idées derrière RSpec, ayant des tests qui ressemblent à des descriptions de comportement en anglais.

Vous voudrez peut-être lire la section sur les métadonnées RSpec avant de lire cette section car, en fin de compte, le filtrage RSpec est basé sur les métadonnées RSpec.

Imaginez que vous avez un fichier de spécifications et qu'il contient deux types de tests (exemples): des tests fonctionnels positifs et des tests négatifs (erreur). Définissons-les comme ça -

RSpec.describe "An Example Group with positive and negative Examples" do 
   context 'when testing Ruby\'s build-in math library' do
      
      it 'can do normal numeric operations' do 
         expect(1 + 1).to eq(2) 
      end 
      
      it 'generates an error when expected' do
         expect{1/0}.to raise_error(ZeroDivisionError) 
      end
      
   end 
end

Maintenant, enregistrez le texte ci-dessus dans un fichier appelé 'filter_spec.rb', puis exécutez-le avec cette commande -

rspec filter_spec.rb

Vous verrez une sortie qui ressemble à ceci -

.. 
Finished in 0.003 seconds (files took 0.11201 seconds to load) 
2 examples, 0 failures

Et si nous voulions réexécuter uniquement les tests positifs de ce fichier? Ou seulement les tests négatifs? Nous pouvons facilement le faire avec les filtres RSpec. Changez le code ci-dessus en ceci -

RSpec.describe "An Example Group with positive and negative Examples" do 
   context 'when testing Ruby\'s build-in math library' do
      
      it 'can do normal numeric operations', positive: true do 
         expect(1 + 1).to eq(2) 
      end 
      
      it 'generates an error when expected', negative: true do 
         expect{1/0}.to raise_error(ZeroDivisionError) 
      end
      
   end 
end

Enregistrez vos modifications dans filter_spec.rb et exécutez cette commande légèrement différente -

rspec --tag positive filter_spec.rb

Maintenant, vous verrez une sortie qui ressemble à ceci -

Run options: include {:positive=>true} 
. 
Finished in 0.001 seconds (files took 0.11401 seconds to load) 
1 example, 0 failures

En spécifiant --tag positive, nous demandons à RSpec d'exécuter uniquement des exemples avec la variable de métadonnées: positive définie. Nous pourrions faire la même chose avec des tests négatifs en exécutant la commande comme ceci -

rspec --tag negative filter_spec.rb

Gardez à l'esprit que ce ne sont que des exemples, vous pouvez spécifier un filtre avec le nom de votre choix.

Formateurs RSpec

Les formateurs permettent à RSpec d'afficher la sortie des tests de différentes manières. Créons un nouveau fichier RSpec contenant ce code -

RSpec.describe "A spec file to demonstrate how RSpec Formatters work" do 
   context 'when running some tests' do 
      
      it 'the test usually calls the expect() method at least once' do 
         expect(1 + 1).to eq(2) 
      end
      
   end 
end

Maintenant, enregistrez-le dans un fichier appelé formatter_spec.rb et exécutez cette commande RSpec -

rspec formatter_spec.rb

Vous devriez voir une sortie qui ressemble à ceci -

. 
Finished in 0.002 seconds (files took 0.11401 seconds to load) 
1 example, 0 failures

Maintenant, exécutez la même commande mais cette fois spécifiez un formateur, comme celui-ci -

rspec --format progress formatter_spec.rb

Vous devriez voir la même sortie cette fois -

. 
Finished in 0.002 seconds (files took 0.11401 seconds to load) 
1 example, 0 failures

La raison en est que le formateur «progress» est le formateur par défaut. Essayons ensuite un autre formateur, essayez d'exécuter cette commande -

rspec --format doc formatter_spec.rb

Vous devriez maintenant voir cette sortie -

A spec file to demonstrate how RSpec Formatters work 
   when running some tests 
      the test usually calls the expect() method at least once
Finished in 0.002 seconds (files took 0.11401 seconds to load) 
1 example, 0 failures

Comme vous pouvez le voir, la sortie est assez différente avec le formateur «doc». Ce formateur présente la sortie dans un style de documentation. Vous vous demandez peut-être à quoi ressemblent ces options en cas d'échec d'un test (exemple). Changeons le code enformatter_spec.rb ressembler à ça -

RSpec.describe "A spec file to demonstrate how RSpec Formatters work" do 
   context 'when running some tests' do 
      
      it 'the test usually calls the expect() method at least once' do 
         expect(1 + 1).to eq(1) 
      end
      
   end 
end

L'attente expect(1 + 1).to eq(1)devrait échouer. Enregistrez vos modifications et réexécutez les commandes ci-dessus -

rspec --format progress formatter_spec.rb et rappelez-vous, puisque le formateur «progress» est la valeur par défaut, vous pouvez simplement exécuter: rspec formatter_spec.rb. Vous devriez voir cette sortie -

F 
Failures:
1) A spec file to demonstrate how RSpec Formatters work when running some tests 
the test usually calls the expect() method at least once
   Failure/Error: expect(1 + 1).to eq(1)
	
      expected: 1
         got: 2
			  
      (compared using ==)			  
   # ./formatter_spec.rb:4:in `block (3 levels) in <top (required)>'

Finished in 0.016 seconds (files took 0.11201 seconds to load)
1 example, 1 failure
Failed examples:

rspec ./formatter_spec.rb:3 # A spec file to demonstrate how RSpec 
   Formatters work when running some tests the test usually calls 
   the expect() method at least once

Maintenant, essayons le formateur de documents, exécutez cette commande -

rspec --format doc formatter_spec.rb

Maintenant, avec le test échoué, vous devriez voir cette sortie -

A spec file to demonstrate how RSpec Formatters work
   when running some tests
      the test usually calls the expect() method at least once (FAILED - 1)
		
Failures:

1) A spec file to demonstrate how RSpec Formatters work when running some
   tests the test usually calls the expect() method at least once
   Failure/Error: expect(1 + 1).to eq(1)
	
   expected: 1
        got: 2
		  
   (compared using ==)
   # ./formatter_spec.rb:4:in `block (3 levels) in <top (required)>'
	
Finished in 0.015 seconds (files took 0.11401 seconds to load) 
1 example, 1 failure

Exemples d'échec

rspec ./formatter_spec.rb:3 # Un fichier de spécifications pour montrer comment les formateurs RSpec fonctionnent lors de l'exécution de certains tests, le test appelle généralement la méthode expect () au moins une fois.

Les formateurs RSpec offrent la possibilité de changer la façon dont les résultats des tests s'affichent, il est même possible de créer votre propre formateur personnalisé, mais c'est un sujet plus avancé.

Lorsque vous apprenez RSpec, vous pouvez lire beaucoup de choses sur les attentes et cela peut être un peu déroutant au début. Il y a deux détails principaux que vous devez garder à l'esprit lorsque vous voyez le terme Attente -

  • Une attente est simplement une déclaration dans un it block qui utilise le expect()méthode. C'est ça. Ce n'est pas plus compliqué que ça. Lorsque vous avez un code comme celui-ci:expect(1 + 1).to eq(2), vous avez une attente dans votre exemple. Vous vous attendez à ce que l'expression1 + 1 évalue à 2. Le libellé est cependant important puisque RSpec est un cadre de test BDD. En appelant cette déclaration une attente, il est clair que votre code RSpec décrit le «comportement» du code qu'il teste. L'idée est que vous exprimez la façon dont le code doit se comporter, d'une manière qui se lit comme de la documentation.

  • La syntaxe Expectation est relativement nouvelle. Avant leexpect() a été introduite (en 2012), RSpec a utilisé une syntaxe différente basée sur le should()méthode. L'attente ci-dessus est écrite comme ceci dans l'ancienne syntaxe:(1 + 1).should eq(2).

Vous pouvez rencontrer l'ancienne syntaxe RSpec pour les attentes lorsque vous travaillez avec un code plus ancien ou une version plus ancienne de RSpec. Si vous utilisez l'ancienne syntaxe avec une nouvelle version de RSpec, vous verrez un avertissement.

Par exemple, avec ce code -

RSpec.describe "An RSpec file that uses the old syntax" do
   it 'you should see a warning when you run this Example' do 
      (1 + 1).should eq(2) 
   end 
end

Lorsque vous l'exécutez, vous obtiendrez une sortie qui ressemble à ceci -

. Deprecation Warnings:

Using `should` from rspec-expectations' old `:should` 
   syntax without explicitly enabling the syntax is deprecated. 
   Use the new `:expect` syntax or explicitly enable 
	
`:should` with `config.expect_with( :rspec) { |c| c.syntax = :should }`
   instead. Called from C:/rspec_tutorial/spec/old_expectation.rb:3 :in 
   `block (2 levels) in <top (required)>'.

If you need more of the backtrace for any of these deprecations to
   identify where to make the necessary changes, you can configure 
`config.raise_errors_for_deprecations!`, and it will turn the deprecation 
   warnings into errors, giving you the full backtrace.

1 deprecation warning total 
Finished in 0.001 seconds (files took 0.11201 seconds to load) 
1 example, 0 failures

À moins que vous ne soyez obligé d'utiliser l'ancienne syntaxe, il est fortement recommandé d'utiliser expect () au lieu de should ().