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 ().