در این آموزش قصد داریم یک پروژهٔ ماشین حساب ساده را با استفاده از یونیت تست تکمیل کنیم و برای آنکه تستهای مرتبط با این پروژه از سایر تستها مجزا باشند، داخل پوشهٔ tests
پوشهٔ دیگری تحت عنوان calculator
ساخته و داخل آن فایلی با نامی دلخواه همچون CalculatorTest.php
به صورت زیر میسازیم:
<?php
use PHPUnit\Framework\TestCase;
class CalculatorTest extends TestCase
{
public function testAddOperands()
{
$calculator = new \App\Calculator;
$calculator->setOperands([5, 10]);
$this->assertEquals(15, $calculator->add());
}
}
همانطور که میبینیم، متدی نوشتهایم تحت عنوان ()testAddOperands
که نام آن با کلمهٔ test
شروع شده که این یک باید است و داخل این متد متغیری تعریف کردهایم به نام calculator$
که مقدارش برابر با آبجکتی از روی کلاسی به نام Calculator
است که هنوز ساخته نشده است.
در ادامه، متدی تحت عنوان ()setOperands
را روی آبجکتی که ساختهایم فراخوانی کرده و به عنوان آرگومان ورودی این متد هم آرایهای حاوی مقادیر ۵ و ۱۰ در نظر گرفتهایم و در نهایت هم متد ()assertEquals
از فریمورک PHPUnit را فراخوانی کرده و دو آرگومان ورودی برایش در نظر گرفتهایم که مورد اول عدد ۱۵ است که به نوعی نتیجهای است که قصد داریم به آن برسیم و آرگومان دوم نیز فراخوانی متد ()add
از کلاس Calculator
است که هنوز نوشته نشده است.
در این مرحله از کار، با ران کردن تست خواهیم دید که اروری در معرض دیدمان قرار خواهد گرفت به طوری که حاوی پیام زیر است:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
.....E 6 / 6 (100%)
Time: 26 ms, Memory: 4.00 MB
There was 1 error:
1) CalculatorTest::testAddOperands
Error: Class 'App\Calculator' not found
/var/www/phpunit/tests/calculator/CalculatorTest.php:8
ERRORS!
میبینیم که در متن ارور آمده است که کلاس Calculator
وجود ندارد که برای رفع این مشکل، داخل پوشهٔ app
فایلی به نام Calculator.php
به صورت زیر میسازیم:
<?php
namespace App;
class Calculator
{
}
با ران کردن مجدد تست خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
.....E 6 / 6 (100%)
Time: 25 ms, Memory: 4.00 MB
There was 1 error:
1) CalculatorTest::testAddOperands
Error: Call to undefined method App\Calculator::setOperands()
/var/www/phpunit/tests/calculator/CalculatorTest.php:9
ERRORS!
Tests: 6, Assertions: 6, Errors: 1.
گرچه تست پاس نشد اما نوع پیام خطا تغییر کرد به طوری که گفته شده متدی با نام ()setOprands
وجود ندارد که برای رفع این ارور، کلاس Calculator
را به صورت زیر تکمیل میکنیم:
<?php
namespace App;
class Calculator
{
public $operands;
public function setOperands(array $operands)
{
$this->operands = $operands;
}
}
داخل این کلاس یک پراپرتی داریم به نام operands$
که این پراپرتی داخل متدی به نام ()setOperands
مقداردهی خواهد شد. در واقع، این متد یک پارامتر ورودی تحت عنوان operands$
از جنس آرایه میگیرد که در نهایت این پارامتر به پراپرتی تعریفشده در بدنهٔ کلاس منتسب میگردد. به درج کلیدواژهای همچون array
پیش از نام پارامتر ورودی این متد اصطلاحاً Type Hinting گفته میشود که این تضمین را ایجاد میکند که پارامترهای ورودی این متد حتماً میباید از جنس آرایه باشند که در این ارتباط و برای کسب اطلاعات بیشتر، میتوانید به مقاله آشنایی با مفهوم Type Hinting در زبان PHP مراجعه نمایید. حال اگر تست را ران کنیم خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
.....E 6 / 6 (100%)
Time: 25 ms, Memory: 4.00 MB
There was 1 error:
1) CalculatorTest::testAddOperands
Error: Call to undefined method App\Calculator::add()
/var/www/phpunit/tests/calculator/CalculatorTest.php:10
ERRORS!
Tests: 6, Assertions: 6, Errors: 1.
میبینیم که متن خطا مجدد تغییر کرد مبنی بر اینکه متدی تحت عنوان ()add
وجود ندارد که این متد را نیز به صورت زیر پیادهسازی میکنیم:
public function add()
{
return array_sum($this->operands);
}
متد ()array_sum
با توجه به آرگومان ورودیاش که همان پراپرتی operands$
است مقداری را بازمیگرداند که آن مقدار توسط دستور return
به عنوان خروجی این متد در نظر گرفته خواهد شد. در واقع، این متد جمع جبری اِلِمانهای آرایهٔ ورودی را بازمیگرداند. حال اگر تست را ران کنیم، خواهیم دید که با موفقیت پاس خواهد شد:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
...... 6 / 6 (100%)
Time: 24 ms, Memory: 4.00 MB
OK (6 tests, 7 assertions)
در ادامه، قصد داریم سه تست دیگر برای اَعمال ریاضیاتی تفریق، تقسیم و ضرب نوشته سپس کلاس Calculator
را به منظور پاس شدن تستها تکمیل نماییم.
به منظور تست کردن قابلیت تفریق این ماشین حساب، تستی تحت عنوان ()testSubtractOperands
نوشته و آن را به صورت زیر تکمیل میکنیم:
public function testSubtractOperands()
{
$calculator = new \App\Calculator;
$calculator->setOperands([100, 10]);
$this->assertEquals(90, $calculator->subtract());
}
با ران کردن تست نیز به عنوان خروجی خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
......E 7 / 7 (100%)
Time: 25 ms, Memory: 4.00 MB
There was 1 error:
1) CalculatorTest::testSubtractOperands
Error: Call to undefined method App\Calculator::subtract()
/var/www/phpunit/tests/calculator/CalculatorTest.php:17
ERRORS!
Tests: 7, Assertions: 7, Errors: 1.
میبینیم که در متن خطا آمده متدی تحت عنوان ()subtract
وجود ندارد که برای رفع این مشکل، چنین متدی را به کلاس Calculator
به صورت زیر میافزاییم:
public function subtract()
{
return $this->operands[0] - $this->operands[1];
}
کاری که داخل این متد انجام دادهایم بدین صورت است که اِلِمان اول پراپرتی یا بهتر بگوییم آرایهٔ operands$
را از اِلِمان دوم این آرایه کسر کردهایم به طوری که اگر تست را اجرا کنیم در خروجی خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
....... 7 / 7 (100%)
Time: 27 ms, Memory: 4.00 MB
OK (7 tests, 8 assertions)
در ادامهٔ تکمیل تستها و به منظور سنجش عملیات تقسیم ماشین حساب، تست دیگری تحت عنوان ()testDivideOperands
به صورت زیر مینویسیم:
public function testDivideOperands()
{
$calculator = new \App\Calculator;
$calculator->setOperands([100, 2]);
$this->assertEquals(50, $calculator->divide());
}
در این تست دو عملوند ۱۰۰ و ۲ را در نظر گرفته سپس در متد ()assertEquals
گفتهایم که تقسیم عدد ۱۰۰ بر عدد ۲ میباید ۵۰ گردد؛ سپس به عنوان پارامتر دوم این متد، متدی تحت عنوان ()divide
را به آبجکت calculator$
منتسب کردهایم. در این مرحله از کار تستمان پاس نخواهد شد چرا که متد ()divide
تعریف نشده است که برای رفع این مشکل چنین متدی داخل کلاس Calculator
به صورت زیر تعریف میکنیم:
public function divide()
{
return $this->operands[0] / $this->operands[1];
}
همچون عملیات تفریق، این بار با استفاده از علامت /
اِلِمان اول آرایهٔ operands$
را بر اِلِمان دوم آن تقسیم کردهایم به طوری که از این پس به عنوان خروجی تست خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
........ 8 / 8 (100%)
Time: 25 ms, Memory: 4.00 MB
OK (8 tests, 9 assertions)
در این مرحله از آموزش به افزودن تست آخر میرسیم که مرتبط با عملیات ضرب است به طوری که تستی برای این منظور به صورت زیر اضافه میکنیم:
public function testMultiplyOperands()
{
$calculator = new \App\Calculator;
$calculator->setOperands([10, 2]);
$this->assertEquals(20, $calculator->multiply());
}
همانطور که ملاحظه میشود، چنین تستی قرار است این اطمینان را به ما بدهد که اگر اعدادی همچون ۱۰ و ۲ به متدی تحت عنوان ()multiply
پاس داده شوند، این متد آنها را در یکدیگر ضرب کرده و حاصلضرب آنها برابر با ۲۰ خواهد شد اما این در حالی است که اگر اکنون تست را ران کنیم با اروری مواجه میشویم مبنی بر اینکه چنین متدی وجود ندارد که در همین راستا آن را داخل کلاس Calculator
به صورت زیر تعریف میکنیم:
public function multiply()
{
return $this->operands[0] * $this->operands[1];
}
همانطور که میبینیم از علامت *
به منظور ضرب کردن اِلِمانهای آرایه در یکدیگر استفاده نمودهایم به طوری که اگر اکنون تست را ران کنیم، خروجی زیر در معرض دیدمان قرار خواهد گرفت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
......... 9 / 9 (100%)
Time: 34 ms, Memory: 4.00 MB
OK (9 tests, 10 assertions)
میبینیم که توانستیم با موفقیت چهار تست برای چهار عمل اصلی بنویسیم و تمامی آنها با موفقیت تست را پشت سر گذاشتند.
درآمدی بر نحوهٔ تست کردن اِکسپشنها
گاهی پیش میآید که در عملیات تقسیم یک عدد همچون ۱۰۰ را بر ۰ تقسیم کنیم که در چنین شرایطی مفسر پیاچپی اِکسپشنی را در معرض دیدمان قرار خواهد داد که برای رفع این مشکل، تست جدیدی به صورت زیر مینویسیم تا این موضوع را بیازماییم:
public function testDivisionByZeroThrowsException()
{
$this->expectException('InvalidArgumentException');
$calculator = new \App\Calculator;
$calculator->setOperands([100, 0]);
$calculator->divide();
}
در فریمورک PHPUnit به تابعی دسترسی داریم تحت عنوان ()expectException
که این امکان را در اختیارمان میگذارد تا مشخص کنیم در صورت بروز مشکل، انتظار چه نوع اِکسپشنی را داشته باشیم. برای همین منظور، نام اِکسپشنی همچون InvalidArgumentException
را به عنوان آرگومان ورودی این متد در نظر گرفتهایم سپس همچون تست ()testDivideOperands
، اقدام به سِت کردن عملوندها سپس فراخوانی متد ()divide
نمودهایم و اکنون اگر تست را ران کنیم، با خروجی زیر مواجه خواهیم شد:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
.........E 10 / 10 (100%)
Time: 25 ms, Memory: 4.00 MB
There was 1 error:
1) CalculatorTest::testDivisionByZeroThrowsException
Division by zero
/var/www/phpunit/app/Calculator.php:28
/var/www/phpunit/tests/calculator/CalculatorTest.php:47
ERRORS!
Tests: 10, Assertions: 10, Errors: 1.
میبینیم که پیام خطای «Division by zero» ارائه شده است. حال برای رفع این مشکل، به کلاس Calculator
بازگشته و متد ()divide
آن را به صورت زیر ریفکتور میکنیم:
public function divide()
{
if ($this->operands[1] == 0) {
throw new \InvalidArgumentException("Division by zero is not possible.");
}
return $this->operands[0] / $this->operands[1];
}
ابتدا یک بار تست را ران کرده سپس به بررسی کدهای جدید خواهیم پرداخت به طوری که در خروجی خواهیم داشت:
/var/www/phpunit$ ./vendor/bin/phpunit
PHPUnit 8.1.4 by Sebastian Bergmann and contributors.
Runtime: PHP 7.3.4-1+ubuntu18.04.1+deb.sury.org+3
Configuration: /var/www/phpunit/phpunit.xml
.......... 10 / 10 (100%)
Time: 26 ms, Memory: 4.00 MB
OK (10 tests, 11 assertions)
میبینیم که تستمان با موفقیت پاس شد. در تفسیر تغییراتی که داخل متد ()divide
انجام دادهایم میتوان گفت که با استفاده از یک دستور شرطی چک کردهایم ببینیم که آیا اِلِمان دوم آرایهٔ operands$
برابر با ۰ است یا خیر که اگر اینگونه بود، کلاس InvalidArgumentException
زبان پیاچپی را به عنوان اِکسپشن فراخوانی میکنیم و داخل متد ()testDivisionByZeroThrowsException
نیز گفتیم که اگر اِکسپشن ایجادشده از جنسِ InvalidArgumentException
بود، تست پاس گردد.
ریفکتور کردن تستها به منظو بهبود سورسکد
حال که توانستیم تستها را با موفقیت بنویسیم، به منظور بهبود کلاس CalculatorTest
، در ادامه یکسری تغییرات در این کلاس خواهیم داد به طوری که داریم:
<?php
use App\Calculator;
use PHPUnit\Framework\TestCase;
class CalculatorTest extends TestCase
{
private $calculator;
protected function setUp(): void
{
$this->calculator = new Calculator;
}
public function testAddOperands()
{
// $calculator = new \App\Calculator;
$this->calculator->setOperands([5, 10]);
$this->assertEquals(15, $this->calculator->add());
}
public function testSubtractOperands()
{
// $calculator = new \App\Calculator;
$this->calculator->setOperands([100, 10]);
$this->assertEquals(90, $this->calculator->subtract());
}
public function testDivideOperands()
{
// $calculator = new \App\Calculator;
$this->calculator->setOperands([100, 2]);
$this->assertEquals(50, $this->calculator->divide());
}
public function testMultiplyOperands()
{
// $calculator = new \App\Calculator;
$this->calculator->setOperands([10, 2]);
$this->assertEquals(20, $this->calculator->multiply());
}
public function testDivisionByZeroThrowsException()
{
$this->expectException('InvalidArgumentException');
// $calculator = new \App\Calculator;
$this->calculator->setOperands([100, 0]);
$this->calculator->divide();
}
}
همانطور که پیش از این دیدیم، به منظور دستیابی به کلاس Calculator
، داخل تکتک تستها یک آبجکت جدید از روی این کلاس ساختیم و این در حالی است که به شکل بهینهتری میتوان این کار را انجام داد بدین صورت که یک پراپرتی در بدنهٔ کلاس از جنس private
تحت عنوان calculator$
ساخته سپس متدی تحت عنوان ()setUp
نوشتهایم که داخل فریمورک PHPUnit تعبیه شده است و همانطور که از علائم void :
مشخص است، این متد قرار نیست چیزی را بازگرداند. داخل این متد هم دستور دادهایم تا یک آبجکت جدید از روی کلاس Calculator
ساخته شده و به پراپرتی calculator$
منتسب گردد و بر خلاف روال گذشته، این بار به منظور ساخت آبجکت به جای درج نام App\Calculator\
صرفاً نام کلاس را نوشته اما در خط دوم فایل آن را use
کردهایم. از این پس، به جای آنکه داخل تکتک تستها یک آبجکت جدید از روی کلاس Calculator
بسازیم، به سادگی میتوانیم با استفاده از دستور <-this$
به پراپرتی calculator$
دست پیدا کنیم.
به خاطر داشته باشید |
زمانی که میخواهیم به یک پراپرتی در داخل متدهای یک کلاس دست پیدا کنیم، هرگز نباید پس از دستور <-this$ علامت $ مرتبط را پراپرتی را بنویسیم. |
همچنین نکتهٔ بسیار مهمی که در ارتباط با متد ()setUp
وجود دارد اینکه متد مذکور به هر تعداد که تست داخل کلاسمان داشته باشیم، به همان تعداد فراخوانی میشود. به عبارت دیگر، در حال حاضر که چهار تست داخل این کلاس نوشتهایم، چهار بار این متد فراخوانی شده و چهار آبجکت مختلف از روی کلاس Calculator
ساخته میشود و همین مسئله این تضمین را ایجاد میکند که هر تست با یک آبجکت دستنخوره سروکار داشته باشد.
همانطور که در این آموزش دیدیم، با استفاده از سَبک برنامهنویسی TDD به سادگی توانستیم ابتدا اقدام به نوشتن متدهای تست چهار عمل اصلی یک ماشین حساب کرده سپس دست به کدنویسی فانکشنهای اصلی ماشین حساب زدیم تا جایی که در نهایت کلیهٔ تستها با موفقیت پاس شدند. همچنین دیدیم که به چه شکل میتوان با استفاده از فریمورک PHPUnit اقدام به تست کردن اِکسپشنها نمود.