You can write very quickly unit tests for the accessors, adders and removers of your entities.
- PHP : 7.2 or later
- phpunit/phpunit : 8 or later
- With Composer
composer require zackad/phpunit-entity-tester --dev
This is an example of unit test file:
<?php
namespace Demo\Tests;
use PHPUnit\Framework\TestCase;
use Zackad\PhpUnitEntityTester\AccessorTester;
use Zackad\PhpUnitEntityTester\AccessorCollectionTester;
class MyEntityTest extends TestCase
{
public function simpleTest()
{
$entity = new MyEntity(); // The entity that you want to test
// Test 'setName' and 'getName'
(new AccessorTester($entity, 'name'))->test('John Doe');
// Test 'addFruit', 'removeFruit' and 'getFruits'
(new AccessorCollectionTester($entity, 'fruits'))->test('apple', 'pear');
}
}
use Zackad\PhpUnitEntityTester\AccessorTester;
$tester = new AccessortTester($entity, 'attribute');
Where $entity
is the entity to test and 'attribute'
the base name of setter and getter methods.
In this case, the tester will test the methods 'setAttribute' and 'getAttribute'.
Your can change the setter and getter methods like this :
$tester->setterMethod('setFoo') //fluent method
->getteMethod('getBar'); // fluent method
You can remove the fluent constraint for the setter (set to true
by default)
$tester->fluent(false);
Simply use the test
method :
$tester->test('the value');
In this case the tester :
- calls the setter with 'the value'
- tests the fluent constraint
- tests that the getter return 'the value'
If the value returned by the getter have to be different to the value used with the setter
use the second argument of the test
method like this :
$tester->test('value for setter', 'value that the getter have to return');
You can also only test the setter method like that :
$tester->testSetter('value');
This method calls the setter method and tests the fluent constraint
You can also only test the setter method like that :
$tester->testGetter('value');
This method calls the getter method and tests the return with value in parameter
use Zackad\PhpUnitEntityTester\AccessorCollectionTester;
$tester = new AccessorCollectionTester($entity, 'items');
Where $entity
is the entity to test and 'items'
the base name for adder, remover and getter methods. For the adder and remover methods, the final 's' of the base name will be removed.
In this case, the tester will test the methods 'addItem', 'removeItem' and 'getItems'.
You can change the adder, remover and getter methods like this :
$tester->addMethod('addCustomItem') // fluent method
->removeMethod('popItem') // fluent method
->getMethod('getAllItems'); // fluent method
You can remove the fluent constraint for the adder and remover methods (set to true
by default)
$tester->fluent(false);
By default, the tester consider that the collection respect the unicity of its items. You can force the tester to consider that the collection don't respect the unicity like that :
$tester->unique(false);
Simply use the test
method :
$tester->test('first value', 'second value');
This method needs two arguments to work. In this case the tester :
- calls and tests the adder method with the first value
- try to remove the second value (that is not in collection yet)
- calls and tests the adder method with the second value
- try to add again the first value (to the unique constraint)
- try to remove the first value
- add again the first value (to obtain a collection with the first and the second value in it)
This method uses the following three methods that you can also use separately.
You can also only test the adder method like that :
$tester->testAdd('value');
This method calls the adder method and tests :
- the fluent constraint
- if the value is in collection (added)
- the unicity constraint
You can also only test the remover method like that :
$tester->testRemove('value');
This method calls the remover method and tests :
- the fluent constraint
- if the value is not in collection (removed)
- if the remover removed the good number of items (all that match value but not more)
You can also only test the getter method like that :
$tester->testGet();
This method tests if the returned value of the getter method is :
- (if return is an object) implement
Countable
interface - (if return is an object) implement
Traversable
interface - (if return is not an object) is an
array
ornull
To test entity quicker, use dataProvider
with tester.
- Create the test method
/**
* @dataProvider providerTestAccessor
*/
public function testAccessor($attribute, $setValue, $getValue = AccessorTester::USE_SET_DATA)
{
$entity = new YourEntityClass();
$tester = new AccessorTester($entity, $attribute);
$tester->test($setValue, $getValue);
}
- Create the data provider
public function providerTestAccessor()
{
return [
['name', 'John Doe'],
['surname', 'JD'],
['astro', 'lion'],
['age', 12],
['age', -5, 0],
['color', 'red', '#ff0000'],
['createdAt', '2015-12-01', new \Datetime('2015-12-01')],
// ...
// one line here, generate complete test of your accessor
];
}
You can also use the process for AccessorCollectionTester
by customize the test method.
Special thanks for gerg0ire who encourage me to do that library