2021-01-31 11:05:37 +01:00
|
|
|
# PHP Unit Test
|
|
|
|
|
|
|
|
## Installation & Configuration
|
|
|
|
|
|
|
|
### Dev-Only Installation
|
|
|
|
|
|
|
|
```ps1
|
|
|
|
composer require --dev phpunit/phpunit
|
|
|
|
```
|
|
|
|
|
|
|
|
```json
|
|
|
|
"require-dev": {
|
|
|
|
"phpunit/phpunit": "<version>"
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Config
|
|
|
|
|
|
|
|
PHPUnit can be configured in a XML file called `phpunit.xml`:
|
|
|
|
|
|
|
|
```xml
|
|
|
|
<phpunit xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
|
|
|
xsi:noNamespaceSchemaLocation="vendor/phpunit/phpunit/phpunit.xsd"
|
|
|
|
bootstrap="vendor/autoload.php"
|
|
|
|
colors="true">
|
|
|
|
|
|
|
|
<testsuites>
|
|
|
|
<testsuit name="App\\Tests">
|
|
|
|
<directory>./test<directory>
|
|
|
|
</testsuit>
|
|
|
|
</testsuites>
|
|
|
|
|
|
|
|
<filter>
|
|
|
|
<whitelist processUncoveredFilesFromWhitelist="true">
|
|
|
|
<directory suffix=".php">./src</directory>
|
|
|
|
</whitelist>
|
|
|
|
</filter>
|
|
|
|
</phpunit>
|
|
|
|
```
|
|
|
|
|
|
|
|
## Testing
|
|
|
|
|
|
|
|
### Test Structure
|
|
|
|
|
2021-07-12 16:18:53 +02:00
|
|
|
**PHPUnit** tests are grouped in classes suffixed with `Test`. Each class *extends* `PHPUnit\Framework\TestCase`.
|
2021-09-20 19:35:32 +02:00
|
|
|
A test is a method of a *test class* prefixed with `test`.
|
2021-01-31 11:05:37 +01:00
|
|
|
PHPUnit is executed from the command line with `vendor/bin/phpunit --colors`.
|
|
|
|
|
|
|
|
```php
|
|
|
|
namespace App;
|
|
|
|
|
|
|
|
class Filter
|
|
|
|
{
|
|
|
|
public function isEmail(string $email): bool
|
|
|
|
{
|
|
|
|
// @todo implement
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```php
|
|
|
|
namespace App\Test;
|
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
use App\Filter;
|
|
|
|
|
|
|
|
class FilterTest extends TestCase
|
|
|
|
{
|
|
|
|
public function testValidMail()
|
|
|
|
{
|
|
|
|
$filter = new Filter();
|
|
|
|
$this->assertTrue($filter->isEmail("foo@bar.com"));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testInvalidEmail()
|
|
|
|
{
|
|
|
|
$filter = new Filter();
|
|
|
|
$this->assertFalse($filter->idEmail("foo"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### [PHPUnit Assertions](https://phpunit.readthedocs.io/en/9.3/assertions.html)
|
|
|
|
|
|
|
|
- `asseretTrue()`: verifies that the element is true
|
|
|
|
- `assertFalse()`: verifies that the element is false
|
|
|
|
- `assertEmpty()`: verifies that the element is empty
|
|
|
|
- `assertEquals()`: verifies that the two elements are equal
|
2021-09-20 19:35:32 +02:00
|
|
|
- `assertGreaterThan()`: verifies that the element is greater than ...
|
2021-01-31 11:05:37 +01:00
|
|
|
- `assertContains()`: verifies that the element is contained in an array
|
|
|
|
- `assertInstanceOf()`: verifies that the element is an instance of a specific class
|
2021-09-20 19:35:32 +02:00
|
|
|
- `assertArrayHasKey(mixed $key, array $array)`: verify that a specific key is in the array
|
2021-01-31 11:05:37 +01:00
|
|
|
|
|
|
|
### [PHPUnit Testing Exceptions](https://phpunit.readthedocs.io/en/9.3/writing-tests-for-phpunit.html#testing-exceptions)
|
|
|
|
|
|
|
|
```php
|
|
|
|
public function testAggiungiEsameException(string $esame)
|
|
|
|
{
|
|
|
|
$this->expectException(Exception::class);
|
|
|
|
$this->expectExceptionMessage("exception_message");
|
|
|
|
|
2021-09-20 19:35:32 +02:00
|
|
|
// execute code that should throw an exception
|
2021-01-31 11:05:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// https://github.com/sebastianbergmann/phpunit/issues/2484#issuecomment-648822531
|
2021-09-20 19:35:32 +02:00
|
|
|
public function testExceptionNotThrown()
|
2021-01-31 11:05:37 +01:00
|
|
|
{
|
2021-09-20 19:35:32 +02:00
|
|
|
$exceptionWasThrown = false;
|
2021-01-31 11:05:37 +01:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// code that should succeed
|
|
|
|
}
|
|
|
|
catch (EsameException $e)
|
|
|
|
{
|
2021-09-20 19:35:32 +02:00
|
|
|
$exceptionWasThrown = true;
|
2021-01-31 11:05:37 +01:00
|
|
|
}
|
|
|
|
|
2021-09-20 19:35:32 +02:00
|
|
|
$this->assertFalse($exceptionWasThrown);
|
2021-01-31 11:05:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// same as
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @doesNotPerformAssertions
|
|
|
|
*/
|
2021-07-12 16:18:53 +02:00
|
|
|
public function testNoExceptions(string $esame)
|
2021-01-31 11:05:37 +01:00
|
|
|
{
|
2021-07-12 16:18:53 +02:00
|
|
|
// code that should succeed (exceptions will make the test fail)
|
2021-01-31 11:05:37 +01:00
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2021-07-12 16:18:53 +02:00
|
|
|
### Test Setup & Teardown (Example)
|
2021-01-31 11:05:37 +01:00
|
|
|
|
|
|
|
```php
|
|
|
|
class ClassTest extends TestCase
|
|
|
|
{
|
|
|
|
// initialize the test
|
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
file_put_contents("/tmp/foo", "Test")
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset the test
|
|
|
|
public function tearDown(): void
|
|
|
|
{
|
|
|
|
unlink("/tmp/foo")
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFoo()
|
|
|
|
{
|
|
|
|
// use temp file
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2022-08-06 10:48:24 +02:00
|
|
|
> **Note**: `setUp()` and `tearDown()` are called *before* and *after* each test method.
|
2021-01-31 11:05:37 +01:00
|
|
|
|
|
|
|
### Data Provider
|
|
|
|
|
|
|
|
```php
|
|
|
|
class DataTest extends TestCase
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @dataProvider provider
|
|
|
|
*/
|
|
|
|
public function testAdd($a, $b, $expected)
|
|
|
|
{
|
|
|
|
$this->assertEquals($expected, $a + $b);
|
|
|
|
}
|
|
|
|
|
2021-09-20 19:35:32 +02:00
|
|
|
// test receives array contents as input
|
2021-01-31 11:05:37 +01:00
|
|
|
public function provider()
|
|
|
|
{
|
|
|
|
// must return array of arrays
|
|
|
|
return [
|
|
|
|
[0, 0, 0],
|
|
|
|
[0, 1, 1]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2021-09-20 19:35:32 +02:00
|
|
|
// test receives array of arrays as input
|
2021-01-31 11:05:37 +01:00
|
|
|
public function provideArrayOfArrays()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
[
|
|
|
|
[
|
|
|
|
[0, 0, 0],
|
|
|
|
[0, 1, 1]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Mock Objects
|
|
|
|
|
|
|
|
```php
|
|
|
|
class UnitTest extends TestCase
|
|
|
|
{
|
|
|
|
public function setUp()
|
|
|
|
{
|
|
|
|
// names of mock are independent from tested class variables
|
|
|
|
$this->mock = $this->createMock(ClassName::class); // create a mock object of a class
|
|
|
|
$this->returned = $this->createMock(ClassName::class); // mock of returned object
|
|
|
|
|
|
|
|
$this->mock->method("methodName") // simulate method on mock
|
|
|
|
->with($this->equalTo(param), ...) // specify input params (one param per equalTo)
|
|
|
|
->willReturn($this->returned); // specify return value
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testMethod()
|
|
|
|
{
|
|
|
|
$this->mock
|
|
|
|
->method("methodName")
|
|
|
|
->with($this->equalTo($arg)) // arg passed to the method
|
|
|
|
->willReturn(value); // actual return value for THIS case
|
|
|
|
// or
|
|
|
|
->will($this->throwException(new Exception())); // method will throw exception
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Code Coverage (needs [XDebug](https://xdebug.org/))
|
|
|
|
|
|
|
|
```ps1
|
2021-09-20 19:35:32 +02:00
|
|
|
vendor/bin/phpunit --coverage-text # code coverage analysis in the terminal
|
2021-01-31 11:05:37 +01:00
|
|
|
```
|