php unit ve tdd

Post on 10-May-2015

3.159 Views

Category:

Documents

0 Downloads

Preview:

Click to see full reader

TRANSCRIPT

PHPUnit

Mustafa İlerİ

TEMMUZ'13 -PHPISt

Test Driven Development

Mustafa İLERİ

@mustafailerihttp://tr.linkedin.com/in/mustafailerimustafa.ileri@sonsuzdongu.comhttp://sonsuzdongu.com/blog

TEST ODAKLI GELİŞTİRME

Kısa gelİştİrme süreçlerİNİN tekrarlarına dayalı bİr yazılım gelİştİrme sürecİdİr.TEST ODAKLI Gelİştİrme sürecİnİn temelİnde testler vardır.Kent Beck tarafından ortaya atılmıştır.

TEST ODAKLI GELİŞTİRME Sürecİ

TESTLERİNİZİ YAZIN.

Testlerİ ÇALIŞTIRIP TEST SONUÇLARININ BAŞARISIZ OLDUĞUNU GÖRÜN (KIRMIZI).

TESTİN GEREKSİNİMLERİNİ YERİNE GETİREN METODLARI YAZIN.

TEKRAR TESTLERİ ÇALIŞTIRIN VE TESTLERİN BAŞARILI OLDUĞUNU GÖRÜN (YEŞİL).

KODUNUZU DÜZENLEYİN.

AYNI SÜRECİ TEKRAR EDİN.

1.2.3.4.5.6.

BİRİM (UNIT)GELİŞTİRİLEN YAZILIMIN Test edİLEBİLİR EN KÜÇÜK PARÇASIDIR.

BİRİM TESTLERİ (UNIT TESTS)

BİRİM TESTİ, YAZILIM GelİŞTİRMEDE BİR YAZILIM TASARIMI VE GELİŞTİRME YÖNTEMİDİR.

BU YÖNTEMDE YAZILIMCI, YAZILIM KODUNU OLUŞTURAN BİRİMLERİN KULLANIMA HAZIR OLDUĞUNA İKNA OLUR.

NEDENNEDEN bİRİM TEST YAZMALIYIZ? bİRİM TEST YAZMALIYIZ?

KOD KAYNAKLI SORUNLARI DAHA RAHAT BULABİLMEK.KOD KAYNAKLI SORUNLARI DAHA RAHAT BULABİLMEK.KOD ÜZERİNDE DAHA RAHAT DEĞİŞİKLİK YAPABİLMEYİ SAĞLAMAK.KOD ÜZERİNDE DAHA RAHAT DEĞİŞİKLİK YAPABİLMEYİ SAĞLAMAK.

ENTEGRASYON VE KULLANICI TESTİ SÜREÇLERİNİ DAHA RAHAT GEÇİREBİLMEK.ENTEGRASYON VE KULLANICI TESTİ SÜREÇLERİNİ DAHA RAHAT GEÇİREBİLMEK.YAZDIĞIMIZ METODLARIN (BİRİMLERİN) KULLANIM ÖRNEĞİNİ SAĞLAMAK.YAZDIĞIMIZ METODLARIN (BİRİMLERİN) KULLANIM ÖRNEĞİNİ SAĞLAMAK.

DAHA İYİ KOD YAZABİLMEK.DAHA İYİ KOD YAZABİLMEK.

KODUN ÇALIŞIP ÇALIŞMADIĞINI BİZ ÇOK İYİ BİLİRİZ!KODUN ÇALIŞIP ÇALIŞMADIĞINI BİZ ÇOK İYİ BİLİRİZ!

YeŞİlİ koruyalımKodunuzu göndermeden önce mutlaka tüm testleri çalıştırın.

Whenever you are tempted to type something into Whenever you are tempted to type something into a print statement or a debugger expression, write a print statement or a debugger expression, write it as a test instead.it as a test instead.

Martin FowlerMartin Fowler

bİRİM TEST ÖZELLİKLERİ

➔ HER BİR TEST BİR ÜNİTEYİ (METODU) TEST ETMELİDİR.➔ BİRİM TESTLERİN BAĞIMLILIKLARI OLMAMALIDIR.➔ KOLAY YAZILABİLİR, OKUNABİLİR VE ÇALIŞTIRILABİLİR OLMADIR.➔ BİRİM TEST YAPARKEN GEREKSİNİMLERİ VE TESTE VERİLECEK CEVABI BİLİNİYOR

OLMALIDIR (WHITE BOX TESTING).

ÇOK MU ÇOK MU ZAMANZAMAN ALIYOR? ALIYOR?

MICROSOFT VE IBM'İN YAPTIĞI ARAŞTIRMAYA GÖRE TEST YAZMAK,MICROSOFT VE IBM'İN YAPTIĞI ARAŞTIRMAYA GÖRE TEST YAZMAK,

GELİŞTİRME SÜRESİNİ GELİŞTİRME SÜRESİNİ %15 - %35%15 - %35 UZATIRKEN UZATIRKEN

PROJEDEKİ HATA (BUG) SAYISINI PROJEDEKİ HATA (BUG) SAYISINI %40 - %90%40 - %90 AZALTIYOR. AZALTIYOR.

PHPUNITPHPUNIT

SEBASTIAN BERGMANNSEBASTIAN BERGMANN TARAFINDAN PHP İLE GELİŞTİRİLMİŞ BİR BİRİM TEST “FRAMEWORK”ÜDÜR. XUNIT AİLESİNDEN TÜREMİŞTİR.

https://github.com/sebastianbergmann/phpunit/

http://phpunit.de/manual/3.8/en/installation.htmlhttp://phpunit.de/manual/3.8/en/installation.html

PHPUNIT GEREKSİNİMLERİ VE YÜKLEME

PHPUNIT 3.7 GEREKSİNİMLERMIN PHP 5.3.3 ÖNERİLEN PHP 5.4.7

PHPUNIT 3.8 GEREKSİNİMLERPHP 5.4.7

YÜKLEME

PEARpear config-set auto_discover 1pear install pear.phpunit.de/PHPUnit

COMPOSER{ "require": { "phpunit/phpunit": "3.8.*" }, "config": { "bin-dir": "/usr/local/bin/" }}

PHARwget http://pear.phpunit.de/get/phpunit.pharchmod +x phpunit.phar

temel kuralLAR

1. Test sınıfları “PHPUnit_Framework_TestCase” sınıfından türetilmelidir.

2. Test sınıfları “Test“ kelimesi ile biltmelidir.(FooBarTest)

3. Test sınıflarının isimleri ile bulunduğu dosyaların isimleri aynı olmalıdır.(FooBarTest.php)

4. Test metodları “test“ kelimesi ile başlamalıdır (testDemoMethod) veya test “annotation“ı kullanılmalıdır.

5. Test metodları “public“ olmalıdır.

6. Test sınıfları en az 1 adet test metodu içermelidir. Aksi durumda test başarısız olur.

7. Test metodları en az 1 adet onaylama (assertion) içermelidir.

İLK İLK TESTTEST

class DemoTest extends PHPUnit_Framework_TestCaseclass DemoTest extends PHPUnit_Framework_TestCase{{ public function testDemo()public function testDemo() {{ $this->assertTrue(true);$this->assertTrue(true); }}}}

ONAYLAMA İFADELERİ, BELİRLİ PARAMETRELER İLE TEST EDİLEN BİRİMİN VERDİĞİ DÖNÜŞ DEĞERİNİ KONTROL ETMEYE YARAR.

public function testArrayHasKey(){

$this->assertArrayHasKey('foo', array('bar' => 'baz'));}

Failed asserting that an array has the key 'foo'.

FAIL!

public function testArrayHasKey(){

$this->assertArrayHasKey('bar', array('bar' => 'baz'));}

SUCCESS!

ONAYLAMA İFADELERİ (ASSERTIONS)

assertArrayHasKey()assertClassHasAttribute()assertClassHasStaticAttribute()assertContains()assertContainsOnly()assertContainsOnlyInstancesOf()assertCount()assertEmpty()assertEqualXMLStructure()assertEquals()assertFalse()assertFileEquals()assertFileExists()assertGreaterThan()assertGreaterThanOrEqual()assertInstanceOf()assertInternalType()assertJsonFileEqualsJsonFile()assertJsonStringEqualsJsonFile()assertJsonStringEqualsJsonString()

assertLessThan()assertLessThanOrEqual()assertNull()assertObjectHasAttribute()assertRegExp()assertStringMatchesFormat()assertStringMatchesFormatFile()assertSame()assertSelectCount()assertSelectEquals()assertSelectRegExp()assertStringEndsWith()assertStringEqualsFile()assertStringStartsWith()assertTag()assertThat()assertTrue()assertXmlFileEqualsXmlFile()assertXmlStringEqualsXmlFile()AssertXmlStringEqualsXmlString()

ONAYLAMA İFADELERİ (ASSERTIONS)PHPUNIT, FARKLI DURUMLARIN DOĞRULUĞUNU KONTROL ETMEK İÇİN ÇOK SAYIDA

DOĞRULAMA İFADESİ SAĞLAR.

http://phpunit.de/manual/3.8/en/writing-tests-for-phpunit.html#writing-tests-for-phpunit.assertions

ONAYLAMA İFADELERİ (ASSERTIONS)KENDİ DOĞRULAMA İFADELERİNİZİ YAZABİLİRSİNİZ.

/** * Assert response code * * @param int $code * @param string $message * @return void */ public function assertResponseCode($code, $message = '') { $this->_incrementAssertionCount(); require_once 'Zend/Test/PHPUnit/Constraint/ResponseHeader.php'; $constraint = new Zend_Test_PHPUnit_Constraint_ResponseHeader(); $response = $this->response; if (!$constraint->evaluate($response, __FUNCTION__, $code)) { $constraint->fail($response, $message); } }

BİR TEST İÇİNDE BİR TEST İÇİNDE BİRDEN FAZLA ONAYLAMA İFADESİBİRDEN FAZLA ONAYLAMA İFADESİ KULLANABİLİRSİNİZ. KULLANABİLİRSİNİZ.TESTİN BAŞARILI OLABİLMESİ İÇİN TESTİN BAŞARILI OLABİLMESİ İÇİN BÜTÜN ONAYLAMA İFADELERİ SAĞLANMALIDIRBÜTÜN ONAYLAMA İFADELERİ SAĞLANMALIDIR..

ONAYLAMA İFADELERİ (ONAYLAMA İFADELERİ (ASSERTIONSASSERTIONS))

AnnotationsPHPUNIT, “annotatıon” KULLANARAK AÇIKLAMA SATIRINDA TEST METODU İÇİN

BELLİ DEKLARASYONLAR YAPMAYI DESTEKLER.

@author@backupGlobals@backupStaticAttributes@codeCoverageIgnore*@covers@coversNothing@dataProvider@depends@expectedException@expectedExceptionCode

@expectedExceptionMessage@group@outputBuffering@preserveGlobalState@requires@runTestsInSeparateProcesses@runInSeparateProcess@test@testdox@ticket

http://phpunit.de/manual/3.8/en/appendixes.annotations.html

DATA PROVIDER“@dataprovider providerMethodName” şeklinde belirtilir ve test

için kullanılacak verileri dizi olarak sağlar.

class DataTest extends PHPUnit_Framework_TestCase{ /** * @dataProvider provider */ public function testAdd($a, $b, $c) { $this->assertEquals($c, $a + $b); } public function provider() { return array( array(0, 0, 0), array(0, 1, 1), array(1, 0, 1), array(1, 1, 3) ); }

1) DataProviderTest::testAdd with data set #3 (1, 1, 3)Failed asserting that 2 matches expected 3.

Depends“@depends dependedMethodName” şeklinde belirtilir ve test

metodları arasındaki bağımlılıkları belirtir.

class DependencyTest extends PHPUnit_Framework_TestCase{ public function testEmpty() { $stack = array(); $this->assertEmpty($stack);

return $stack; }

/** * @depends testEmpty */ public function testPush(array $stack) { array_push($stack, 'foo'); $this->assertEquals('foo', $stack[count($stack)-1]); $this->assertNotEmpty($stack);

return $stack; }}

EXPECTED EXCEPTION / CODE“@expectedException NotFoundException” şeklinde kullanılır ve beklenilen hata türü belirlenir.

“@expectedExceptionCode 100” şeklinde kullanılır ve beklenilen hata kodu belirlenir.

“@expectedExceptionMessage Right Message” şeklinde kullanılır ve beklenilen hata mesajı belirlenir.

class ExceptionTest extends PHPUnit_Framework_TestCase{ /** * @expectedException InvalidArgumentException * @expectedExceptionCode 20 * @expectedExceptionMessage Right Message */ public function testExceptionHasRightCode() { throw new InvalidArgumentException('Right Message', 20); }}

TAKLİT NESNELERİ TAKLİT NESNELERİ ((MOCK OBJECTSMOCK OBJECTS))

Testİn Yazıldığı BİRİMDE bulunan bağımlı Testİn Yazıldığı BİRİMDE bulunan bağımlı nesneler GİBİ davranan taklİt nesnelerİDİR.nesneler GİBİ davranan taklİt nesnelerİDİR.

TEMEL AMAÇ; TEMEL AMAÇ; DIŞA BAĞIMLILIĞI KALDIRMAKDIŞA BAĞIMLILIĞI KALDIRMAK VE VE TESTLERİN İZOLASYONUNUTESTLERİN İZOLASYONUNU SAĞLAMAK. SAĞLAMAK.

GENELDE VERİTABANI, MAıL CLIENT VE WEB GENELDE VERİTABANI, MAıL CLIENT VE WEB SERVİSLER İÇİN KULLANILIR.SERVİSLER İÇİN KULLANILIR.

TAKLİT NESNELERİ (TAKLİT NESNELERİ (MOCK OBJECTSMOCK OBJECTS))<?phpclass Service_User{ ... public function setUserModel($model) { $this->_userModel = $model; } ... public function getBankAccountByUserId($userId) { $bankAccount = $this->_userModel->getBankAccountByUserId($userId); (is_array($bankAccount) && sizeof($bankAccount > 0)) ? $this->setServiceResponseData($bankAccount) : $this->setServiceResponseError ('bankAccount', 'Bank account cannot be found.');

return $this->getServiceResponse(); } ...

TAKLİT NESNELERİ (MOCK OBJECTS)<?php

class Model_User{

... public function getBankAccountByUserId($userId) { $select = $this->db()->select() ->from(array('BA' => 'bankAccount')) ->join(array('B' => 'bank'), 'B.id = BA.bankId') ->where('userId = ?', $userId); return $this->db()->fetchRow($select); } ...}

TAKLİT NESNELERİ (TAKLİT NESNELERİ (MOCK OBJECTSMOCK OBJECTS))<?php

class UserServiceTest extends Zend_Test_PHPUnit_ControllerTestCase{ public function testGetBankAccountByUserId() { $userService = new Service_User(); $mockOfModel = $this->getMock('Model_User'); $mockOfModel->expects($this->once()) ->method('getBankAccountByUserId') ->with(0) ->will($this->returnValue(array(1, 2, 3, 4, 5)));

$userService->setUserModel($mockOfModel); $serviceResponse = $userService->getBankAccountByUserId(0); if ($serviceResponse->getSuccess()) { $this->assertTrue(is_array($serviceResponse->getData())); } else { $this->assertTrue(false); } }}

PHPUNIT EKLENTİLERİ

DBUNITpear install phpunit/DbUnit"phpunit/dbunit": ">=1.2"

Veritabanı etkileşimlerini test etmeye yarar.

INVOKERpear install phpunit/PHP_Invoker"phpunit/php-invoker": "*"

Zaman aşımı durumlarını test etmeye yarar.

SELENIUMpear install phpunit/PHPUnit_Selenium"phpunit/phpunit-selenium": ">=1.2"

PHPUnit için Selenium RC entegrasyonu sağlar.

XHProfpear install phpunit/PHPUnit_TestListener_XHProf

Test edilen kodu otomatik olarak analiz eder.

http://phpunit.de/manual/3.8/en/installation.html

Command Lıne Tool

Bir yapılandırma dosyanız var ise, testler phpunit komutu ile çalıştırılır.Yapılandırma dosyanız yok ise yardım için phpunit yazmanız yeterli.

phpunit DependencyTestphpunit DependencyTest.php

➜ tests phpunit DependencyTest PHPUnit 3.7.19 by Sebastian Bergmann.

...SFI.E.

Time: 0 seconds, Memory: 5.50Mb

OK (3 tests, 5 assertions)

(.) ➜ Test başarılı(F) ➜ Test başarısız(S) ➜ Test değerlendirmeye alınmadı(I) ➜ Test henüz tamamlanmamış $this->markTestIncomplete();(E) ➜ Hata oluşma durumu

http://phpunit.de/manual/3.8/en/textui.html

Code Coverage

Code Coverage, yazılan testlerin, projenizdeki kodların ne kadarını kapsadığını ölçümler ve raporlar.

Fakat bunun için xDebug'ın kurulu olması gerekmektedir.

PHPUnit, bu raporu HTML olarak oluşturabilir.

Code Coverage sürekli yapılan geliştirmelerde kodunuzun güvenirliği üzerindeki değişimi rahatça görebilmenizi sağlar.

Code Coverage

Code Coverage

Code Coverage

Code Coverage

Phpunıt YapılandırmasıTestlerin genel yapılandırması bir xml dosyası ile belirlenebilir.

testsuites: Test paketleri oluşturulmasını sağlar.groups: Testlere dahil edilecek veya çıkarılacak gruplar belirlenir.filter: Code Coverage'a dahil edilecek veya hariç tutulacak birimler belirlenebilir.

phpunit.xml

<phpunit bootstrap="./bootstrap.php" colors="true"> <testsuite name="Application Test Suite"> <directory>./application</directory> </testsuite> <testsuite name="Library Test Suite"> <directory>./library</directory> </testsuite> <filter> <blacklist> <directory suffix=".php">/usr/lib/php/pear/Zend</directory> </blacklist> </filter></phpunit>

http://phpunit.de/manual/3.8/en/appendixes.configuration.html

Test tİPLERİ

BİRİM TEST

ENTEGRASYON TESTİ

SİSTEM TESTİ

KULLANICI KABUL TESTİ

➜➜

ENTEGRASYON TESTİ

ASLINDA BİRİM TESTİN BİR TÜRÜDÜR.BİRİM TEST, BİRİMİ TEST EDERKEN; ENTEGRASYON TESTİ BERABER ÇALIŞAN BİRİMLERİ, MODÜLLERİ TEST EDER.

1. BİLEŞEN TESTİ: SADECE BİR BİLEŞENİ TEST EDER. DIŞA BAĞIMLILIKLAR İÇİN TAKLİT NESNELERİ (MOCK OBJECTS) KULLANILIR.

2. ENTEGRASYON TESTİ: BİR VEYA BİRDEN FAZLA BİLEŞENİ TEST EDER. İHTİYAÇ DUYULMASI DURUMUNDA DIŞA BAĞIMLI BİLEŞENLERE ERİŞEBİLİR.

SİSTEM TESTİ

YAZILIM VE DONANIM SİSTEMLERİNİN ENTEGRE EDİLMESİYLE, SİSTEMİN BEKLENEN ŞARTLARA UYGUNLUĞUNUN TEST EDİLMESİDİR.

YAZILIM PERFORMAS TESTİYÜK TESTİUYUMLULUK TESTİSTRES TESTİÖLÇEKLENEBİLİRLİK TESTİVS...

KULLANICI KABUL TESTİ

GELİŞTİRİLEN YAZILIMın, SON KULLANICININ BEKLENTİLERİNİ KARŞILAYIP KARŞILAYAMADIĞINI TEST EDER.

ÖNCEDEN TANIMLANMIŞ TEST SENARYOLARININ DOĞRULUĞUNU KONTROL EDer.KULLANICI KABUL TESTİ, TEST UZMANLARI TARAFINDAN GERÇEKLEŞTİRİLİR.

SON KULLANICI TESTLERİ BEHAT VEYA SELENIUM KULLANARAK OTOMATİZE EDİLEBİLİR.

kaynaklar

http://www.slideshare.net/lemiorhan/the-engines-of-software-development-testing-and-test-driven-developmenthttp://www.slideshare.net/markstory/phpunit-and-you?from_search=10http://www.slideshare.net/mjlivelyjr/automated-unit-testing?from_search=38http://www.slideshare.net/ferca_sl/unit-testing-with-phpunit?from_search=47http://www.cihataltuntas.com/test-driven-development/http://en.wikipedia.org/wiki/Test-driven_developmenthttp://en.wikipedia.org/wiki/Software_testing

Resimlerhttp://diogoosorio.com/blog/entry/test-driven-development-tdd-using-phpunithttp://elberling.dk/advertising-illustrations/crash-test-dummy-head-038/http://www.thesaleslion.com/massive-blog-growth-havetime-takes/http://amyshirateitel.com/2011/12/17/vintage-space-fun-fact-gene-kranzs-vest/http://toutelaculture.com/arts/folk-rock-et-protest-songs-dylan-sexpose-a-la-cite-de-la-musique/http://www.ahawallpaper.com/the-ring-two/1280x1024-0-773500.htmlhttp://www.movievortex.com/2009/11/fight-clubs-blu-ray-debut-contains-surprise/http://commons.wikimedia.org/wiki/File:Martin_Fowler_-_Swipe_Conference_2012_.jpghttp://www.freegreatpicture.com/green-sky/grass-material-15815

sorularsorular

top related