Slide 1

Slide 1 text

UNE RÉVOLUTION DANS LE MONDE DES TESTS UNE RÉVOLUTION DANS LE MONDE DES TESTS Boris Feld - PyConfr 2018 octobus.net

Slide 2

Slide 2 text

PRESENTATION PRESENTATION

Slide 3

Slide 3 text

BORIS FELD BORIS FELD Expert Mercurial, Python et DevOps. Enthousiaste sur la qualité. https://octobus.net/ @lothiraldan

Slide 4

Slide 4 text

QUI FAIT DES TESTS? QUI FAIT DES TESTS?

Slide 5

Slide 5 text

SONDAGE: COMBIEN DE PERSONNES SONDAGE: COMBIEN DE PERSONNES ÉCRIVENT DES TESTS ? ÉCRIVENT DES TESTS ?

Slide 6

Slide 6 text

SONDAGE: COMBIEN DE PERSONNES SONDAGE: COMBIEN DE PERSONNES EXECUTENT DES TESTS ? EXECUTENT DES TESTS ?

Slide 7

Slide 7 text

VOUS N’ÊTES PAS LES SEULS VOUS N’ÊTES PAS LES SEULS 2017 2018 0.00% 10.00% 20.00% 30.00% 40.00% 50.00% 60.00% 70.00% 80.00% Launching tests Writing tests => 70% lancent des tests

Slide 8

Slide 8 text

ÉTAT DE L’ART EN PYTHON ÉTAT DE L’ART EN PYTHON

Slide 9

Slide 9 text

OUTILS OUTILS Unittest Nosetests Pytest Autre ?

Slide 10

Slide 10 text

UNITTEST UNITTEST Inclut dans Python Permet d’écrire des tests sous la forme de classes Fournit des assertions sous la forme de méthodes d’assertions.

Slide 11

Slide 11 text

UNITTEST EXEMPLE UNITTEST EXEMPLE import unittest class TestStringMethods(unittest.TestCase): def test_upper(self): self.assertEqual('foo'.upper(), 'FOO') def test_isupper(self): self.assertTrue('FOO'.isupper()) self.assertTrue('Foo'.isupper()) if __name__ == '__main__': unittest.main()

Slide 12

Slide 12 text

EXECUTION AVEC UNITTEST EXECUTION AVEC UNITTEST F. ================================================================= ===== FAIL: test_isupper (__main__.TestStringMethods) ----------------------------------------------------------------- ----- Traceback (most recent call last): File "test.py", line 10, in test_isupper self.assertTrue('Foo'.isupper()) AssertionError: False is not true ----------------------------------------------------------------- ----- Ran 2 tests in 0.000s FAILED (failures=1)

Slide 13

Slide 13 text

NOSETESTS NOSETESTS Un unittest meilleur Un ecosystem de plugin Une vraie avancée Mais deux paquets Pypi, nose et nose2 Un écosystème parcemé

Slide 14

Slide 14 text

EXECUTION AVEC NOSETESTS EXECUTION AVEC NOSETESTS F. ================================================================= ===== FAIL: test_isupper (test.TestStringMethods) ----------------------------------------------------------------- ----- Traceback (most recent call last): File "/home/lothiraldan/Labo/presentations/balto_revolution_fr/src/tes t.py", line 10, in test_isupper self.assertTrue('Foo'.isupper()) AssertionError: False is not true ----------------------------------------------------------------- ----- Ran 2 tests in 0.000s FAILED (failures=1)

Slide 15

Slide 15 text

REDNOSE (PLUGIN NOSE1) REDNOSE (PLUGIN NOSE1) F. ================================================================= ===== 1) FAIL: test_isupper (test.TestStringMethods) ----------------------------------------------------------------- ----- Traceback (most recent call last): test.py line 10 in test_isupper self.assertTrue('Foo'.isupper()) AssertionError: False is not true ----------------------------------------------------------------- ------------ 2 tests run in 0.001 seconds. 1 FAILED (1 test passed)

Slide 16

Slide 16 text

PYTEST PYTEST La vraie étoile montante du marché Un meilleur unittest Tests sous forme de fonctions Fixtures De nombreux utilitaires De nombreux plugins

Slide 17

Slide 17 text

EXECUTION AVEC PYTEST EXECUTION AVEC PYTEST ============================= test session starts ============================== platform linux2 -- Python 2.7.15, pytest-3.1.2, py-1.5.3, pluggy- 0.4.0 rootdir: /home/lothiraldan/Labo/presentations/balto_revolution_fr/src, inifile: collected 2 items test.py F. =================================== FAILURES =================================== ________________________ TestStringMethods.test_isupper ________________________ self = def test_isupper(self): self.assertTrue('FOO'.isupper()) > self.assertTrue('Foo'.isupper())

Slide 18

Slide 18 text

PYTEST-SUGAR PYTEST-SUGAR

Slide 19

Slide 19 text

QU’EST-CE QU’ON VEUT DANS UNE QU’EST-CE QU’ON VEUT DANS UNE INTERFACE ? INTERFACE ?

Slide 20

Slide 20 text

MA LISTE AU PÈRE NOËL MA LISTE AU PÈRE NOËL Couleur Barre de progression Lancer uniquement les tests échoués Lancer un test donné Interface web

Slide 21

Slide 21 text

BALTO BALTO

Slide 22

Slide 22 text

BALTO KESAKO? BALTO KESAKO? Ce n’est pas YAPTL (Yet Another Python Testing Library).

Slide 23

Slide 23 text

COULEUR COULEUR

Slide 24

Slide 24 text

BARRE DE PROGRESSION BARRE DE PROGRESSION

Slide 25

Slide 25 text

SÉLECTIONNER SEULEMENT LES TESTS QUI SÉLECTIONNER SEULEMENT LES TESTS QUI ONT ECHOUES ONT ECHOUES

Slide 26

Slide 26 text

SÉLECTIONNER SEULEMENT UN TEST SÉLECTIONNER SEULEMENT UN TEST

Slide 27

Slide 27 text

BALTO DEMO BALTO DEMO [Demo live]

Slide 28

Slide 28 text

ARCHITECTURE ARCHITECTURE

Slide 29

Slide 29 text

QUESTION POUR UN CHAMPION QUESTION POUR UN CHAMPION C’est quoi Balto ? Un plugin pytest Une interface web Un serveur websocket Un lanceur de processus

Slide 30

Slide 30 text

LA BONNE RÉPONSE LA BONNE RÉPONSE

Slide 31

Slide 31 text

TOUTES CES RÉPONSES TOUTES CES RÉPONSES Pytest Pytest-plugin Balto-Server Balto-UI

Slide 32

Slide 32 text

COMMUNICATION COMMUNICATION Pytest Pytest-plugin Process Balto-Server ? Balto-UI WS

Slide 33

Slide 33 text

LE SECRET DE BALTO LE SECRET DE BALTO

Slide 34

Slide 34 text

LE SECRET LE SECRET Le Secret de Balto c’est qu’il lit la sortie standard du processus lancé

Slide 35

Slide 35 text

L’AUTRE PARTIE DU SECRET L’AUTRE PARTIE DU SECRET Mais il faut que le plugin, balto-server et balto-ui puissent se comprendre

Slide 36

Slide 36 text

POUR SE COMPRENDRE POUR SE COMPRENDRE Il faut une langue commun

Slide 37

Slide 37 text

SORTIE STANDARD DE PYTEST SORTIE STANDARD DE PYTEST ============================= test session starts ============================== platform linux2 -- Python 2.7.15, pytest-3.1.2, py-1.5.3, pluggy- 0.4.0 rootdir: /home/lothiraldan/Labo/presentations/balto_revolution_fr/src, inifile: collected 2 items test.py F. =================================== FAILURES =================================== ________________________ TestStringMethods.test_isupper ________________________ self = def test_isupper(self): self.assertTrue('FOO'.isupper()) > self.assertTrue('Foo'.isupper()) E AssertionError: False is not true

Slide 38

Slide 38 text

LES FORMATS DE SORTIE DE TEST LES FORMATS DE SORTIE DE TEST Junit.xml Tap Mozlog

Slide 39

Slide 39 text

JUNIT.XML JUNIT.XML Basé sur XML Un gros chier à la n du build Format lié à Junit => Non streamable

Slide 40

Slide 40 text

TAP TAP Très connu dans la communauté Perl Très simpliste, peu extensible Format lié à TAP Perl => Nécessite un parseur en Python et en JS

Slide 41

Slide 41 text

MOZLOG MOZLOG Format utilisé dans les tests Mozilla 1 au début et à la n de chaque test => Avoir une atomicite, 1 test == 1 ligne est plus simple

Slide 42

Slide 42 text

LISTE AU PÈRE NOËL LISTE AU PÈRE NOËL Simple à écrire et lire Streaming pour pas attendre la n du run Format indépendant d’une implémentation

Slide 43

Slide 43 text

UN NOUVEAU FORMAT UN NOUVEAU FORMAT

Slide 44

Slide 44 text

DE QUOI ON AS BESOIN? DE QUOI ON AS BESOIN? Nom du test Le résultat du test (failed, passed, skipped…) Error en cas d’erreur

Slide 45

Slide 45 text

EXEMPLE EXEMPLE { "test_name": "test_success", "outcome": "passed" } { "test_name": "test_failed", "outcome": "failed", "error": { "humanrepr": "def test_fails():\n> assert False\nE assert False\n\ntest_func.py:9: AssertionError" } }

Slide 46

Slide 46 text

PIÈCE MANQUANTE PIÈCE MANQUANTE { "test_number": 6, "_type": "session_start" } { "failed": 3, "total_duration": 0.03, "_type": "session_end", "skipped": 0, "error": 0, "passed": 3 }

Slide 47

Slide 47 text

PLUS DE DONNÉES PLUS DE DONNÉES Temps Messages de log Sortie standard et sortie standard d’erreur Diff de text ou d’images Fichier, ligne, etc…

Slide 48

Slide 48 text

LANCER UN TEST SPECIFIQUE? LANCER UN TEST SPECIFIQUE?

Slide 49

Slide 49 text

SORTIE VS ENTRÉE SORTIE VS ENTRÉE On as aussi besoin d’un format d’entrée: { "collect_only": true } { "files": ["file1.py", "file2.py"], "nodes": ["file1.py::test_func"] }

Slide 50

Slide 50 text

COMMENT RÉCUPÉRER UN IDENTIFIANT? COMMENT RÉCUPÉRER UN IDENTIFIANT? Ajoutons un nouveau champ de sortie: { "test_name": "test_success", "outcome": "passed", "_id": "test_func.py::test_success" }

Slide 51

Slide 51 text

LA POST-RÉVOLUTION LA POST-RÉVOLUTION

Slide 52

Slide 52 text

EXEMPLE DE SORTIE EXEMPLE DE SORTIE { "test_name": "test_failed", "_id": "test_func.py::test_success", "outcome": "failed", "error": { "humanrepr": "def test_fails():\n> assert False\nE assert False\n\ntest_func.py:9: AssertionError" } }

Slide 53

Slide 53 text

ET SI ON ALLAIT PLUS LOIN ? ET SI ON ALLAIT PLUS LOIN ? Et si on faisait un plugin nosetests pour ce format?

Slide 54

Slide 54 text

ET SI ON ALLAIT ENCORE PLUS LOIN ? ET SI ON ALLAIT ENCORE PLUS LOIN ? Et si on faisait un plugin pour un test runner dans un autre langage?

Slide 55

Slide 55 text

JUSQU’AU PEUT-ON ALLER ? JUSQU’AU PEUT-ON ALLER ? Et si on lisait la sortie standard d’autre chose qu’un process ? Comme une commande ssh ou un container Docker ?

Slide 56

Slide 56 text

VERS L’INFINIE ET AU-DELÀ VERS L’INFINIE ET AU-DELÀ Balto etant base sur Asyncio, et si on lancait plusieurs suites pytest en parallelle ? Ou plusieurs suites dans des langages differents sur une machine distante avec Docker?

Slide 57

Slide 57 text

RÉVOLUTION DEMO RÉVOLUTION DEMO

Slide 58

Slide 58 text

LA VRAIE RÉVOLUTION LA VRAIE RÉVOLUTION

Slide 59

Slide 59 text

UN NOUVEAU PROTOCOLE UN NOUVEAU PROTOCOLE Similaire à HTTP ou LSP, c’est une fondation au dessus duquel construire les nouveaux outils de demain.

Slide 60

Slide 60 text

EXEMPLE EXEMPLE Vous voulez un dashboard curses pour lancer vos tests? Vous pouvez ! Vous voulez un rapport sur les temps de chacun de vos tests? Vous pouvez !

Slide 61

Slide 61 text

LITF LITF Language Independent Test Format

Slide 62

Slide 62 text

BALTO BALTO BAlto is a Language Independent Test Orchestrator

Slide 63

Slide 63 text

INSTALLER BALTO INSTALLER BALTO Télécharger la dernière release sur Github: https://github.com/Lothiraldan/balto/releases Le mettre dans le path Installer le plugin litf de votre test runner Pro ter

Slide 64

Slide 64 text

QUE POUVEZ-VOUS FAIRE ? QUE POUVEZ-VOUS FAIRE ? Utiliser Balto Parler de Balto autour de vous Parler de LITF au mainteneur de votre test runner préféré

Slide 65

Slide 65 text

CONTRIBUER CONTRIBUER Balto et LITF sont open-source : https://github.com/lothiraldan/litf https://github.com/lothiraldan/balto

Slide 66

Slide 66 text

CONCLUSION CONCLUSION

Slide 67

Slide 67 text

QUESTIONS? QUESTIONS?