Introducción a testing en php
-
Upload
ismael-ambrosi -
Category
Technology
-
view
160 -
download
4
Transcript of Introducción a testing en php
Introducción a!Testing
Testing¿Para que sirve?
¿Para que sirve?• Verificar del funcionamiento de la aplicación!
• Automatización!
• Evitar recurrir en bugs!
• Construir casos de uso difíciles de reproducir!
• Simplificar la introducción de cambios!
• Simplificar la refactorización!
¿Que vamos a ver?
• Unit Testing!
• Functional Testing
Unit Testing
Es una forma de probar el correcto funcionamiento de un módulo de código. Esto sirve para asegurar que cada uno de los módulos funcione
correctamente por separado.
http://es.wikipedia.org/wiki/Prueba_unitaria
¿Ventajas?• Facilita el debug!
• Demuestra el funcionamiento del código sin importar el entorno!
• Viendo el test puedo determinar que hace el código!
• Confirma si el pequeño cambio en realidad fue tan pequeño!
• El código tiende a ser más limpio!
• Nos permite corregir bugs y asegurarnos que no vuelven a aparecer!
• Simplifica la integración
¿Desventajas?
• No muestran errores de integración!
• Es difícil anticipar todos los casos de uso y futuros bugs!
• Código extenso
Herramientas
PHPUnit http://phpunit.de/!
SimpleTest http://simpletest.org/!
SnapTest https://github.com/Jakobo/snaptest
Herramientas
PHPUnit http://phpunit.de/!
SimpleTest http://simpletest.org/!
SnapTest https://github.com/Jakobo/snaptest
<?php!!namespace IsmaAmbrosi;!!class Math!{! private $number;!! public function __construct($number)! {! $this->number = $number;! }!! public function getNumber()! {! return $this->number;! }!! public function sum($number)! {! return $this->number += $number;! }!}
<?php!!namespace !class Math!{! private ! public function {! $this }!! public function {! return }!! public function {! return }!}
<?php!!namespace IsmaAmbrosi\Tests;!!use IsmaAmbrosi\Math;!!/**! * Class MathTest! *! * @package IsmaAmbrosi\Tests! */!class MathTest extends \PHPUnit_Framework_TestCase!{! public function testSumWorks()! {! $math = new Math(5);!! $this->assertEquals(5, $math->getNumber());!! $math->sum(2); # Sumo 2 al valor original!! $this->assertEquals(7, $math->getNumber());! }!}
<?php!!namespace !class Math!{! private ! public function {! $this }!! public function {! return }!! public function {! return }!}
<?php!!namespace IsmaAmbrosi\Tests;!!use IsmaAmbrosi\Math;!!/**! * Class MathTest! *! * @package IsmaAmbrosi\Tests! */!class MathTest extends \PHPUnit_Framework_TestCase!{! public function testSumWorks()! {! $math = new Math(5);!! $this->assertEquals(5, $math->getNumber());!! $math->sum(2); # Sumo 2 al valor original!! $this->assertEquals(7, $math->getNumber());! }!}
Clase base para las clases de tests
<?php!!namespace !class Math!{! private ! public function {! $this }!! public function {! return }!! public function {! return }!}
<?php!!namespace IsmaAmbrosi\Tests;!!use IsmaAmbrosi\Math;!!/**! * Class MathTest! *! * @package IsmaAmbrosi\Tests! */!class MathTest extends \PHPUnit_Framework_TestCase!{! public function testSumWorks()! {! $math = new Math(5);!! $this->assertEquals(5, $math->getNumber());!! $math->sum(2); # Sumo 2 al valor original!! $this->assertEquals(7, $math->getNumber());! }!}
Verifica el valor de getNumber() es el
original
<?php!!namespace !class Math!{! private ! public function {! $this }!! public function {! return }!! public function {! return }!}
<?php!!namespace IsmaAmbrosi\Tests;!!use IsmaAmbrosi\Math;!!/**! * Class MathTest! *! * @package IsmaAmbrosi\Tests! */!class MathTest extends \PHPUnit_Framework_TestCase!{! public function testSumWorks()! {! $math = new Math(5);!! $this->assertEquals(5, $math->getNumber());!! $math->sum(2); # Sumo 2 al valor original!! $this->assertEquals(7, $math->getNumber());! }!}
Verifica que el valor actual sea igual al
resultado de la suma
Métodos comunes$this->assertEquals(2, 1 + 1);!$this->assertTrue(2 == 2);!$this->assertFalse(2 == 4);!$this->assertNull(null);!$this->assertInstanceOf('Namespace\Class', $instance);!$this->assertEmpty("");!$this->assertGreaterThan(2, 5);!$this->assertLessThan(2, 1);!$this->assertCount(3, array(1, 2, 3));!$this->assertFileExists('/path/to/file.txt');
Nota: La falla en el assert finaliza la ejecución del método test.
Instalación
# 1 - Composer!
$ composer require —dev “phpunit/phpunit” “~3.7”
# 2 - Pear!
$ pear config-set auto_discover 1
$pear install pear.phpunit.de/PHPUnit
# 3 - PHP Archive (PHAR)!
$ wget https://phar.phpunit.de/phpunit.phar
Ejecución
$ phpunit
# Single test class!$ phpunit tests/Namespace/ClassTest.php
# Proyecto de Symfony!$ phpunit -c app
Test Driven Development
Es un proceso de desarrollo que se basa en la repetición de un corto ciclo de desarrollo: Primero se desarrollan los test, definiendo como debe
funcionar la librería a desarrollar, luego se escribe la mínima cantidad de código necesaria para que pasen los tests, y finalmente se hace la
refactorización del código a los estandares aceptables.
http://en.wikipedia.org/wiki/Test-driven_development
Escritura de test
Escritura del código necesario para pasar el test
Refactorización
Verificación de los tests
Code CoverageIsmaAmbrosiGeneratorBundle
Code CoverageIsmaAmbrosiGeneratorBundle
Mock Objects
Son objetos que imitan el comportamiento de objetos reales de una forma controlada. Se usan para probar a otros objetos en pruebas unitarias que esperan mensajes de una clase en particular para sus métodos, al igual que los diseñadores de autos usan un crash dummy cuando simulan un
accidente.
http://es.wikipedia.org/wiki/Objeto_simulado
Herramientas
PHPUnit Mock Objects http://phpunit.de/ !
Mockery https://github.com/padraic/mockery!
Prophecy https://github.com/phpspec/prophecy
<?php!!namespace IsmaAmbrosi\Logger;!!/**! * Class Logger! *! * @package IsmaAmbrosi\Logger! */!class Logger!{! private $handler;!! public function __construct(HandlerInterface $handler)! {! $this->handler = $handler;! }!! public function log($text)! {! return $this->handler->write($text);! }!}
<?php!!namespace IsmaAmbrosi\Logger;!!/**! * Interface HandlerInterface! *! * @package IsmaAmbrosi\Logger! */!interface HandlerInterface!{! public function write($log);!}
<?php!!namespace !/**! * Class Logger! *! * @package IsmaAmbrosi\Logger! */!class Logger!{! private ! public function {! }!! public function {! return }!}
<?php!!namespace !/**! * Interface HandlerInterface! *! * @package IsmaAmbrosi\Logger! */!interface {! public function }
<?php!!namespace IsmaAmbrosi\Tests\Logger;!!use IsmaAmbrosi\Logger\Logger;!!/**! * Class LoggerTest! *! * @package IsmaAmbrosi\Tests\Logger! */!class LoggerTest extends \PHPUnit_Framework_TestCase!{! public function testLogger()! {! $handler = $this->getMock('IsmaAmbrosi\Logger\HandlerInterface');!! $logger = new Logger($handler);!! $logger->log('Este es un mensaje de log');! }!}
!# src/IsmaAmbrosi/Logger/Logger.php!public function log($text)!{! return $this->handler->write($text);!}!!!# src/IsmaAmbrosi/Tests/Logger/LoggerTest.php!public function testLogger()!{! $handler = $this->getMock('IsmaAmbrosi\Logger\HandlerInterface');! $handler! ->expects($this->any())! ->method('write')! ->will($this->returnValue(7));!! $logger = new Logger($handler);!! $this->assertEquals(7, $logger->log('Message'));!}
Ejemplo
Functional Testing
Una prueba funcional es una prueba basada en la ejecución, revisión y retroalimentación de las funcionalidades previamente diseñadas para el
software. Son pruebas específicas, concretas y exhaustivas para probar y validar que el software hace lo que debe y sobre todo, lo que se ha
especificado.
http://es.wikipedia.org/wiki/Pruebas_funcionales
¿Ventajas?
• Se escriben una sola vez!
• Ofrecen mayor cobertura de casos de uso!
• Puedo ver como se comporta mi aplicación y para casos específicos!
• Sirven como test de integración en algunos casos!
• Confirma si los pequeños cambios en realidad fue tan pequeños
¿Desventajas?
• Es difícil descubrir todos los errores del código!
• No siempre puedo determinar que hace el código!
• Es difícil anticipar todos los casos de uso y futuros bugs
<?php!!namespace IsmaAmbrosi\MyBundle\Tests\Controller;!!use Symfony\Bundle\FrameworkBundle\Test\WebTestCase;!!class DefaultControllerTest extends WebTestCase!{! public function testIndex()! {! $client = static::createClient();!! $crawler = $client->request('GET', '/');!! $this->assertEquals(1, $crawler->filter('h1:contains("Welcome")')->count());! }!}
Ejemplo (Symfony)
Pruebas de Integración
Son aquellas que se realizan en el ámbito del desarrollo de software una vez que se han aprobado las pruebas unitarias. Únicamente se refieren a la
prueba o pruebas de todos los elementos unitarios que componen un proceso, hecha en conjunto, de una sola vez.
http://es.wikipedia.org/wiki/Pruebas_de_integración
Behavior Driven Development
Es una técnica de desarrollo ágil de software que nos permite crear una aplicación describiendo su funcionamiento desde la perspectiva de los
inversores.
http://www.ticketea.com/introduccion-bdd-granada
Herramientas
• Behat http://behat.org!
• PHPUnit http://phpunit.de/
Behat
Un framework PHP para testing de las expectativas de negocios.
BehatSe escriben historias en lenguaje entendible para cualquier humano(Gherkin).!
Esas historias luego se comprueban con el funcionamiento de la aplicación.!
Se separan en features.!
Cada feature desarrolla uno o más escenarios.
FeatureFeature: ls! In order to see the directory structure! As a UNIX user! I need to be able to list the current directory's contents!! Scenario: List 2 files in a directory! Given I am in a directory "test"! And I have a file named "foo"! And I have a file named "bar"! When I run "ls"! Then I should get:! """! bar! foo! """
http://docs.behat.org/quick_intro.html
Feature para WebFeature: Login! Para poder acceder al sistema! Como un usuario registrado! Necesito poder identificarme con un usuario y contraseña!! Scenario: Ingreso al sistema con usuario válido! Given I am on “/login”! When I fill in “username” with “iambrosi”! And I fill in “password” with “12345”! And I press “Ingresar”! Then I should see “Bienvenido Ismael”
Para los tests de funcionalidades web se debe utilizar la extensión Mink y uno de los drivers para navegadores.
http://mink.behat.org/
Para los tests de funcionalidades web se debe utilizar la extensión Mink y uno de los drivers para navegadores.
Goutte Sahi Zombie Selenium Selenium2
http://mink.behat.org/
Resumen
Resumen• Utilizamos Unit Testing para probar pequeños módulos
Resumen• Utilizamos Unit Testing para probar pequeños módulos!
• Utilizamos Integration Testing para probar la union de varios módulos
Resumen• Utilizamos Unit Testing para probar pequeños módulos!
• Utilizamos Integration Testing para probar la union de varios módulos !
• Utilizamos Fuctional Testing para probar el resultado de la ejecución
Resumen• Utilizamos Unit Testing para probar pequeños módulos!
• Utilizamos Integration Testing para probar la union de varios módulos!
• Utilizamos Fuctional Testing para probar el resultado de la ejecución!
• Utilizamos BDD(Behat) para desarrollar los test tal cual es visto en el negocio
¿Preguntas?
¡Gracias!