dev-notes/docs/languages/php/unit-tests.md

231 lines
5.2 KiB
Markdown
Raw Normal View History

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
```