Integration testing library in PHP for databases and other common infrastructure related tests.
It is developed as a set of extensions for PHPUnit that hooks on different events and executes your fixtures.
Currently you can run custom fixtures on the following PHPUnit hooks:
BeforeFirstTest
BeforeTest
AfterTest
AfterLastTest
WIP: AMQP specific test fixtures with WithBeforeTestFixtureName and WithAfterTestFixtureName
PHPUnit
Via composer
composer require --dev hrodic/php-integration-testing
On PHPUnit configuration XML file you must specify the extension with its configuration.
You are able to specify the configuration filename that you will be using. Defaults to .integration-testing.json
<extensions> <extension class="IntegrationTestingPHPUnitRunnerExtensionHandler"> <arguments> <string>.integration-testing.json</string> </arguments> </extension> </extensions>
You also check phpunit-integration.xml.dist example
If you need help with PHPUnit extensions, please refer to the Official Documentation
If you need to test the integration of MySQL or MariaDB, use the PDO driver extension.
It requires configuration parameters that can be found in the json config file.
The most important parameters are DSN, username and password of your database + some fixture path definitions.
Example:
"pdo": { "dsn": "mysql:host=localhost:3306;dbname=test;charset=utf8", "user": "test", "password": "test", "fixtures": { "beforeFirstTest": { "path": "tests/fixtures/before-first-test", "extension": "sql" }, "beforeTest": { "path": "tests/fixtures/before-test", "extension": "sql" }, "afterTest": { "path": "tests/fixtures/after-test" }, "afterLastTest": { "path": "tests/fixtures/after-last-test" } } },
You can also try out AMQP (tested on RabbitMQ) fixtures and operations.
Configure your connectivity and the hook operations using the configuration file.
Notes:
Hook definitions are optional, so just configure the ones you need.
You can only publish messages on beforeFirstTest
and on beforeTest
.
You can purge queues in all four hooks.
file extension of message bodies to publish defaults to json
use routing_key
if you have your exchange configured as direct
. You can define it as empty string if fanout
"amqp": { "host": "localhost", "port": 5672, "user": "test", "password": "test", "vhost": "/", "fixtures": { "beforeFirstTest": { "purgeQueues": [ "before-first-test-queue" ], "publishMessages": [ { "exchange": "test-exchange", "queue": "before-first-test-queue", "routing_key": "before-first-test", "path": "tests/fixtures/before-first-test", "extension": "json" } ] }, "beforeTest": { "purgeQueues": [ "before-test-queue" ], "publishMessages": [ { "exchange": "test-exchange", "queue": "before-test-queue", "routing_key": "before-test", "path": "tests/fixtures/before-test" } ] }, "afterTest": { "purgeQueues": [ "before-test-queue" ] }, "afterLastTest": { "purgeQueues": [] } } }
A PDO fixture is just an SQL file.
All the fixtures located in a specific hook category will be executed in order and inside a transaction.
How you create the SQL and the integrity of the database in each stage is up to you. The library does not force you to follow any convention although is common to setup fixtures at the beginning and clean your mess after each test.
You can create, insert, delete or whatever you configure your user to do. Remember, your testing database must be isolated from any real database!
All four fixture hook types could be placed in the directory that you prefer.
For BeforeTest and AfterTest hooks, which occur in every specific test, you can also provide specific fixtures to be executed just after the generic Before and After fixtures by implementing the interfaces WithBeforeTestFixtureName and/or WithAfterTestFixtureName.
final class YourIntegrationTest extends TestCase implements WithBeforeTestFixtureName, WithAfterTestFixtureName { private const FIXTURE_NAME = 'pdo-integration-test'; public static function getAfterTestFixtureName(): string { return self::FIXTURE_NAME; } public static function getBeforeTestFixtureName(): string { return self::FIXTURE_NAME; } public function testYourRepositoryHere(): void { // arrange // act // assert against real database (your fixtures are already there!) } }
The Extension will check if the methods are defined, and use them to locate subdirectories inside the main BEFORE_TEST_PDO_FIXTURES_PATH and AFTER_TEST_PDO_FIXTURES_PATH directories.
If you take a look onto the tests/fixtures folder, you will see an example on how you can organize your fixtures. You can have multiple SQL files and the extension will read and execute them in order.
├── after-last-test # AFTER_LAST_TEST_PDO_FIXTURES_PATH, executed once, at the end │ └── 01.sql ├── after-test # AFTER_TEST_PDO_FIXTURES_PATH, executed after each test │ ├── 01.sql │ └── pdo-integration-test # executed after each test inside the class that defines this fixture name │ └── 01.sql ├── before-first-test # BEFORE_FIRST_TEST_PDO_FIXTURES_PATH, executed once, at the beginning │ └── 01.sql └── before-test # BEFORE_TEST_PDO_FIXTURES_PATH, executed before each test ├── 01.sql └── pdo-integration-test # executed before each test inside the class that defines this fixture name └── 01.sql
Integration testing requires some infrastructure to be in place.
This library assumes (you can check docker-compose.yml file for inspiration) that you have an accessible database or other infrastructure already in place and the database is created.