در این آموزش قصد داریم یک پروژهٔ ماشین حساب ساده را با استفاده از یونیت تست تکمیل کنیم و برای آنکه تستهای مرتبط با این پروژه از سایر تستها مجزا باشند، داخل پوشهٔ 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 اقدام به تست کردن اِکسپشنها نمود.
