feature #116 Accessing tested application services easily via driver's service container (pamil)

This PR was merged into the 2.1-dev branch.

Discussion
----------

Fixes #86, fixes #110.

TODO:
 - [x] Documentation

Commits
-------

776af6cc38 Describe accessing driver's service container
b05304858c Introduce a service in the test application that exposes driver's service container
e94285dc0b Fix coding standard
This commit is contained in:
Kamil Kokot
2020-04-04 15:52:47 +02:00
committed by GitHub
4 changed files with 192 additions and 9 deletions

View File

@@ -0,0 +1,98 @@
Feature: Accessing driver's service container
Background:
Given a working Symfony application with SymfonyExtension configured
And a Behat configuration containing:
"""
default:
extensions:
Behat\MinkExtension:
base_url: "http://localhost:8080/"
default_session: symfony
sessions:
symfony:
symfony: ~
suites:
default:
contexts:
- App\Tests\SomeContext
"""
And a feature file containing:
"""
Feature:
Scenario:
Given the counter service is zeroed
When I visit the page "/hello-world"
Then the counter service should return 1
"""
And a context file "tests/SomeContext.php" containing:
"""
<?php
namespace App\Tests;
use App\Counter;
use Behat\Behat\Context\Context;
use Behat\Mink\Mink;
use Symfony\Component\DependencyInjection\ContainerInterface;
final class SomeContext implements Context {
private $mink;
private $driverContainer;
public function __construct(Mink $mink, ContainerInterface $driverContainer)
{
$this->mink = $mink;
$this->driverContainer = $driverContainer;
}
/** @Given the counter service is zeroed */
public function counterServiceIsZeroed(): void
{
assert(0 === $this->getCounterService()->get());
}
/** @When I visit the page :page */
public function visitPage(string $page): void
{
$this->mink->getSession()->visit($page);
}
/** @Then the counter service should return :number */
public function counterServiceShouldReturn(int $number): void
{
assert($number === $this->getCounterService()->get());
}
private function getCounterService(): Counter
{
return $this->driverContainer->get('App\Counter');
}
}
"""
Scenario: Accessing a service from driver's service container (manually injected dependencies)
Given a YAML services file containing:
"""
services:
App\Tests\SomeContext:
public: true
arguments:
- '@behat.mink'
- '@behat.driver.service_container'
"""
When I run Behat
Then it should pass
Scenario: Accessing a service from driver's service container (autowired & autoconfigured dependencies)
Given a YAML services file containing:
"""
services:
_defaults:
autowire: true
autoconfigure: true
App\Tests\SomeContext: ~
"""
When I run Behat
Then it should pass

View File

@@ -8,6 +8,7 @@ use Behat\Behat\Context\Context;
use Behat\Mink\Mink; use Behat\Mink\Mink;
use Behat\Mink\Session; use Behat\Mink\Session;
use FriendsOfBehat\SymfonyExtension\Mink\MinkParameters; use FriendsOfBehat\SymfonyExtension\Mink\MinkParameters;
use FriendsOfBehat\SymfonyExtension\ServiceContainer\SymfonyExtension;
use Symfony\Component\BrowserKit\Client; use Symfony\Component\BrowserKit\Client;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface; use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\ContainerBuilder;
@@ -15,6 +16,7 @@ use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\DependencyInjection\Definition; use Symfony\Component\DependencyInjection\Definition;
use Symfony\Component\DependencyInjection\Reference; use Symfony\Component\DependencyInjection\Reference;
use Symfony\Component\HttpKernel\DependencyInjection\Extension; use Symfony\Component\HttpKernel\DependencyInjection\Extension;
use Symfony\Component\HttpKernel\KernelInterface;
final class FriendsOfBehatSymfonyExtensionExtension extends Extension implements CompilerPassInterface final class FriendsOfBehatSymfonyExtensionExtension extends Extension implements CompilerPassInterface
{ {
@@ -22,6 +24,7 @@ final class FriendsOfBehatSymfonyExtensionExtension extends Extension implements
{ {
$this->provideMinkIntegration($container); $this->provideMinkIntegration($container);
$this->registerBehatContainer($container); $this->registerBehatContainer($container);
$this->registerDriverBehatContainer($container);
$container->registerForAutoconfiguration(Context::class)->addTag('fob.context'); $container->registerForAutoconfiguration(Context::class)->addTag('fob.context');
} }
@@ -47,6 +50,27 @@ final class FriendsOfBehatSymfonyExtensionExtension extends Extension implements
$container->setDefinition('behat.service_container', $behatServiceContainerDefinition); $container->setDefinition('behat.service_container', $behatServiceContainerDefinition);
} }
private function registerDriverBehatContainer(ContainerBuilder $container): void
{
$driverKernelDefinition = new Definition(KernelInterface::class, [SymfonyExtension::DRIVER_KERNEL_ID]);
$driverKernelDefinition->setFactory([new Reference('behat.service_container'), 'get']);
$driverKernelDefinition->setPublic(true);
$driverKernelDefinition->setLazy(true);
$driverServiceContainerDefinition = new Definition(ContainerInterface::class);
$driverServiceContainerDefinition->setFactory([$driverKernelDefinition, 'getContainer']);
$driverServiceContainerDefinition->setPublic(true);
$driverServiceContainerDefinition->setLazy(true);
$driverTestServiceContainerDefinition = new Definition(ContainerInterface::class, ['test.service_container']);
$driverTestServiceContainerDefinition->setFactory([$driverServiceContainerDefinition, 'get']);
$driverTestServiceContainerDefinition->setPublic(true);
$driverTestServiceContainerDefinition->setLazy(true);
$container->setDefinition('behat.driver.service_container', $driverTestServiceContainerDefinition);
$container->registerAliasForArgument('behat.driver.service_container', ContainerInterface::class, 'driver container');
}
private function provideBrowserKitIntegration(ContainerBuilder $container): void private function provideBrowserKitIntegration(ContainerBuilder $container): void
{ {
if (!class_exists(Client::class) || !$container->has('test.client')) { if (!class_exists(Client::class) || !$container->has('test.client')) {

View File

@@ -34,7 +34,7 @@ final class SymfonyExtension implements Extension
* Kernel used by Symfony driver to isolate web container from contexts' container. * Kernel used by Symfony driver to isolate web container from contexts' container.
* Container is rebuilt before every request. * Container is rebuilt before every request.
*/ */
private const DRIVER_KERNEL_ID = 'fob_symfony.driver_kernel'; public const DRIVER_KERNEL_ID = 'fob_symfony.driver_kernel';
/** @var bool */ /** @var bool */
private $minkExtensionFound = false; private $minkExtensionFound = false;

View File

@@ -92,12 +92,13 @@ CON
$this->thereIsFile('src/Kernel.php', <<<'CON' $this->thereIsFile('src/Kernel.php', <<<'CON'
<?php <?php
declare(strict_types=1);
namespace App; namespace App;
use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait; use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait;
use Symfony\Component\Config\Loader\LoaderInterface; use Symfony\Component\Config\Loader\LoaderInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\Kernel as HttpKernel; use Symfony\Component\HttpKernel\Kernel as HttpKernel;
use Symfony\Component\Routing\RouteCollectionBuilder; use Symfony\Component\Routing\RouteCollectionBuilder;
@@ -105,11 +106,6 @@ class Kernel extends HttpKernel
{ {
use MicroKernelTrait; use MicroKernelTrait;
public function helloWorld(): Response
{
return new Response('Hello world!');
}
public function registerBundles(): iterable public function registerBundles(): iterable
{ {
return [ return [
@@ -125,17 +121,82 @@ class Kernel extends HttpKernel
'secret' => 'Pigeon', 'secret' => 'Pigeon',
]); ]);
$loader->load(__DIR__ . '/../config/default.yaml');
$loader->load(__DIR__ . '/../config/services.yaml'); $loader->load(__DIR__ . '/../config/services.yaml');
} }
protected function configureRoutes(RouteCollectionBuilder $routes) protected function configureRoutes(RouteCollectionBuilder $routes): void
{ {
$routes->add('/hello-world', 'kernel:helloWorld'); $routes->add('/hello-world', 'App\Controller:helloWorld');
} }
} }
CON CON
); );
$this->thereIsFile('src/Controller.php', <<<'CON'
<?php
declare(strict_types=1);
namespace App;
use Symfony\Component\HttpFoundation\Response;
final class Controller
{
private $counter;
public function __construct(Counter $counter)
{
$this->counter = $counter;
}
public function helloWorld(): Response
{
$this->counter->increase();
return new Response('Hello world!');
}
}
CON
);
$this->thereIsFile('src/Counter.php', <<<'CON'
<?php
declare(strict_types=1);
namespace App;
final class Counter
{
private $counter = 0;
public function increase(): void
{
$this->counter++;
}
public function get(): int
{
return $this->counter;
}
}
CON
);
$this->thereIsFile('config/default.yaml', <<<'YML'
services:
App\Controller:
arguments:
- '@App\Counter'
public: true
App\Counter:
public: false
YML
);
$this->thereIsFile('config/services.yaml', ''); $this->thereIsFile('config/services.yaml', '');
} }