SimpleTest - Prendre le contrôle des tests
vendredi 31 décembre 2004 :: perrick :: Traductions :: un commentaire :: aucun trackbackCette page...
- Le temps est souvent une variable négligée dans les tests.
- Une classe horloge nous permet de modifier le temps.
- Nettoyer le test de groupe.
Pour tester un module de code vous avez besoin d'avoir un contrôle très précis sur son environnement. Si quelque chose change dans les coulisses, par exemple dans un fichier de configuration, alors les tests peuvent échouer de façon inattendue. Il ne s'agirait plus d'un test de code sans équivoque et pourrait vous faire perdre des heures précieuses à la recherche d'erreurs dans un code qui fonctionne. Alors qu'il s'agit d'un problème de configuration qui plante le test en question. Au mieux vos scénarios de test deviennent de plus en plus compliqués afin de prendre en compte toutes les variations possibles.
Contrôler le temps
Il y a souvent beaucoup de variables évidentes qui peuvent affecter un scénario de test unitaire, d'autant plus dans un environnement de développement web dans lequel PHP a ses aises. Parmi celles-ci, on trouve les paramètres de connexion à la base de données et ceux de configuration, les droits de fichier et les ressources réseau, etc. L'échec ou la mauvaise installation de l'un ou l'autre de ces composants cassera la suite de test. Est-ce que nous devons ajouter des tests pour valider l'installation de ces composants ? C'est une bonne idée mais si vous les placez dans les tests du module de code vous aller commencer à encombrer votre code de test avec des détails hors de propos avec la tâche en cours. Ils doivent être placés dans leur propre groupe de test.
Par contre un autre problème reste : nos machines de développement doivent aussi avoir tous les composants système d'installés avant l'éxécution de la suite de test. Et vos tests s'éxécuteront plus lentement.
Devant un tel dilemme, nous créerons souvent des versions enveloppantes des classes qui gèrent ces ressources. Les vilains détails de ces ressources sont ensuite codés une seule fois. J'aime bien appeler ces classes des "classes frontière" étant donné qu'elles existent en bordure de l'application, l'interface entre votre application et le reste du système. Ces classes frontière sont - dans le meilleur des cas - simulées pendant les tests par des versions de simulacre. Elles s'exécutent plus rapidement et sont souvent appelées "bouchon serveur [Ndt : Server Stubs]" ou dans leur forme plus générique "objet fantaisie [Ndt : Mock Objects]". Envelopper et bouchonner chacune de ces ressouces permet d'économiser pas mal de temps.
Un des facteurs souvent négligés reste le temps. Par exemple, pour tester l'expiration d'une session des codeurs vont souvent temporairement en caler la durée à une valeur très courte, disons 2 secondes, et ensuite effectuer un sleep(3) : ils estiment alors que la session a expirée. Sauf que cette opération ajoute 3 secondes à la suite de test : il s'agit souvent de beaucoup de code en plus pour rendre la classe de session aussi malléable. Plus simple serait d'avoir un moyen d'avancer l'horloge arbitrairement. De contrôler le temps.
Une classe horloge
Une nouvelle fois, nous allons effectuer notre conception d'une enveloppe d'horloge via l'écriture de tests. Premièrement nous ajoutons un scénario de test d'horloge dans notre suite de test tests/all_tests.php...
<?php if (! defined('SIMPLE_TEST')) { define('SIMPLE_TEST', 'simpletest/'); } require_once(SIMPLE_TEST . 'unit_tester.php'); require_once(SIMPLE_TEST . 'reporter.php'); require_once('log_test.php'); require_once('clock_test.php'); $test = &new GroupTest('All tests'); $test->addTestCase(new TestOfLogging()); $test->addTestCase(new TestOfClock()); $test->run(new HtmlReporter()); ?>Ensuite nous créons le scénario de test dans un nouveau fichier tests/clock_test.php...
<?php require_once('../classes/clock.php'); class TestOfClock extends UnitTestCase { function TestOfClock() { $this->UnitTestCase('Clock class test'); } function testClockTellsTime() { $clock = new Clock(); $this->assertEqual($clock->now(), time(), 'Now is the right time'); } function testClockAdvance() { } } ?>Notre unique test pour le moment, c'est que notre nouvelle class Clock se comporte comme un simple substitut de la fonction time() en PHP. L'autre méthode tient lieu d'emploi. C'est notre chose à faire en quelque sorte. Nous ne lui avons pas donnée de test parce que ça casserait notre rythme. Nous écrirons cette fonctionnalité de décalage dans le temps une fois que nous serons au vert. Pour le moment nous ne sommes évidemment pas dans le vert...
Fatal error: Failed opening required '../classes/clock.php' (include_path='') in /home/marcus/projects/lastcraft/tutorial_tests/tests/clock_test.php on line 2
<?php class Clock { function Clock() { } function now() { } } ?>De la sorte nous reprenons le cours du code.
All tests
Fail: Clock class test->testclocktellstime->[NULL: ] should be equal to [integer: 1050257362]class Clock { function Clock() { } function now() { return time(); } }Et nous revoici dans le vert...
All tests
Le test d'avancement ressemble à...
class TestOfClock extends UnitTestCase { function TestOfClock() { $this->UnitTestCase('Clock class test'); } function testClockTellsTime() { $clock = new Clock(); $this->assertEqual($clock->now(), time(), 'Now is the right time'); } function testClockAdvance() { $clock = new Clock(); $clock->advance(10); $this->assertEqual($clock->now(), time() + 10, 'Advancement'); } }Le code pour arriver au vert est direct : il suffit d'ajouter un décalage de temps.
class Clock { var $_offset; function Clock() { $this->_offset = 0; } function now() { return time() + $this->_offset; } function advance($offset) { $this->_offset += $offset; } }
Nettoyer le test de groupe
Notre fichier all_tests.php contient des répétitions dont nous pourrions nous débarrasser. Nous devons ajouter manuellement tous nos scénarios de test depuis chaque fichier inclus. C'est possible de les enlever mais avec les précautions suivantes. La classe GroupTest inclue une méthode bien pratique appelée addTestFile() qui prend un fichier PHP comme paramètre. Ce mécanisme prend note de toutes les classes : elle inclut le fichier et ensuite regarde toutes les classes nouvellement créées. S'il y a des filles de TestCase elles sont ajoutées au nouveau test de groupe.
Voici notre suite de test remaniée en appliquant cette méthode...
<?php if (! defined('SIMPLE_TEST')) { define('SIMPLE_TEST', 'simpletest/'); } require_once(SIMPLE_TEST . 'unit_tester.php'); require_once(SIMPLE_TEST . 'reporter.php'); $test = &new GroupTest('All tests'); $test->addTestFile('log_test.php'); $test->addTestFile('clock_test.php'); $test->run(new HtmlReporter()); ?>Les inconvéniants sont les suivants...
- Si le fichier de test a déjà été inclus, aucune nouvelle classe ne sera ajoutée au groupe.
- Si le fichier de test contient d'autres classes reliés à TestCase alors celles-ci aussi seront ajouté au test de groupe.
Vos commentaires et/ou trackbacks
Ajouter un commentaire
Les commentaires pour ce billet sont fermés.