* Reset also fob_symfony.driver_kernel between scenarios
* Make sure reset() for the Mink driver implementation creates new KernelBrowser instances, to achieve consistent reboots of the `fob_symfony.driver_kernel` when making more than one request within a single scenario
This commit is contained in:
@@ -0,0 +1,198 @@
|
||||
Feature: Resetting the driver's service container in the right places
|
||||
|
||||
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 YAML services file containing:
|
||||
"""
|
||||
services:
|
||||
_defaults:
|
||||
autowire: true
|
||||
autoconfigure: true
|
||||
|
||||
App\Tests\SomeContext: ~
|
||||
"""
|
||||
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);
|
||||
}
|
||||
|
||||
/** @When I increment the counter */
|
||||
public function incrementCounter(): void
|
||||
{
|
||||
$this->getCounterService()->increase();
|
||||
}
|
||||
|
||||
/** @Then the counter service should return :number */
|
||||
public function counterServiceShouldReturn(int $number): void
|
||||
{
|
||||
assert($number === $this->getCounterService()->get());
|
||||
}
|
||||
|
||||
/** @Then I should see :content on the page */
|
||||
public function shouldSeeContentOnPage(string $content): void
|
||||
{
|
||||
assert(false !== strpos($this->mink->getSession()->getPage()->getContent(), $content));
|
||||
}
|
||||
|
||||
private function getCounterService(): Counter
|
||||
{
|
||||
return $this->driverContainer->get('App\Counter');
|
||||
}
|
||||
}
|
||||
"""
|
||||
|
||||
Scenario: Driver's service container is reset between scenarios
|
||||
# Regression testing https://github.com/FriendsOfBehat/SymfonyExtension/issues/149
|
||||
Given a feature file containing:
|
||||
"""
|
||||
Feature:
|
||||
Scenario: First pass
|
||||
Given the counter service is zeroed
|
||||
When I increment the counter
|
||||
Then the counter service should return 1
|
||||
Scenario: Second pass
|
||||
Given the counter service is zeroed
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: Driver's service container is reset between requests
|
||||
Given a feature file containing:
|
||||
"""
|
||||
Feature:
|
||||
Scenario:
|
||||
Given the counter service is zeroed
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 1" on the page
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 1" on the page
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: Driver's service container can be prepared before a request is made
|
||||
Given a feature file containing:
|
||||
"""
|
||||
Feature:
|
||||
Scenario:
|
||||
Given the counter service is zeroed
|
||||
And I increment the counter
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 2" on the page
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: Driver's service container is not reset before a request is made, even when another scenario made a request before
|
||||
# Regression testing https://github.com/FriendsOfBehat/SymfonyExtension/issues/149
|
||||
Given a feature file containing:
|
||||
"""
|
||||
Feature:
|
||||
Scenario:
|
||||
Given the counter service is zeroed
|
||||
And I increment the counter
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 2" on the page
|
||||
Scenario:
|
||||
Given the counter service is zeroed
|
||||
And I increment the counter
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 2" on the page
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: Driver's service container can be inspected after a request has been made
|
||||
Given 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
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: When multiple requests are made, the driver's service container is reset, but we can inspect "in between" states
|
||||
Given 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
|
||||
# This will reset the driver's container, so we will see "1" again
|
||||
When I visit the page "/hello-world"
|
||||
Then the counter service should return 1
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
|
||||
Scenario: Driver's service container can not reasonably be modified between requests
|
||||
# This is not really a feature, but rather documenting current behavior (judge yourself).
|
||||
# One way around it might be to (how?) disable the reboot feature on the KernelBrowser and
|
||||
# take responsibility for resetting the driver's container yourself.
|
||||
Given a feature file containing:
|
||||
"""
|
||||
Feature:
|
||||
Scenario:
|
||||
Given the counter service is zeroed
|
||||
And I increment the counter
|
||||
When I visit the page "/hello-world"
|
||||
Then I should see "The counter value is 2" on the page
|
||||
And the counter service should return 2
|
||||
# Now a second request is made, which will reset the container, but leaves us
|
||||
# with no easy way of pre-setting the container once again:
|
||||
When I increment the counter
|
||||
# ... you might expect "3"
|
||||
And I visit the page "/hello-world"
|
||||
# ... now you might expect "4". But, in fact, the reset happened just before the request.
|
||||
Then I should see "The counter value is 1" on the page
|
||||
And the counter service should return 1
|
||||
"""
|
||||
When I run Behat
|
||||
Then it should pass
|
||||
@@ -6,35 +6,72 @@ namespace FriendsOfBehat\SymfonyExtension\Driver;
|
||||
|
||||
use Behat\Mink\Driver\BrowserKitDriver;
|
||||
use Symfony\Component\BrowserKit\AbstractBrowser;
|
||||
use Symfony\Component\BrowserKit\Client;
|
||||
use Symfony\Component\HttpKernel\KernelInterface;
|
||||
|
||||
final class SymfonyDriver extends BrowserKitDriver
|
||||
{
|
||||
/** @var KernelInterface */
|
||||
private $kernel;
|
||||
|
||||
/** @var string|null */
|
||||
private $baseUrl;
|
||||
|
||||
public function __construct(KernelInterface $kernel, ?string $baseUrl)
|
||||
{
|
||||
if (!$kernel->getContainer()->has('test.client')) {
|
||||
$this->kernel = $kernel;
|
||||
$this->baseUrl = $baseUrl;
|
||||
|
||||
if (!$this->kernel->getContainer()->has('test.client')) {
|
||||
throw new \RuntimeException(sprintf(
|
||||
'Kernel "%s" used by Behat with "%s" environment and debug %s does not have "test.client" service. ' . "\n" .
|
||||
'Please make sure the kernel is using "test" environment or have "framework.test" configuration option enabled.',
|
||||
get_class($kernel),
|
||||
$kernel->getEnvironment(),
|
||||
$kernel->isDebug() ? 'enabled' : 'disabled',
|
||||
get_class($this->kernel),
|
||||
$this->kernel->getEnvironment(),
|
||||
$this->kernel->isDebug() ? 'enabled' : 'disabled',
|
||||
));
|
||||
}
|
||||
|
||||
/** @var object $testClient */
|
||||
$testClient = $kernel->getContainer()->get('test.client');
|
||||
parent::__construct($this->createBrowser(), $this->baseUrl);
|
||||
}
|
||||
|
||||
if (!$testClient instanceof Client && !$testClient instanceof AbstractBrowser) {
|
||||
public function reset()
|
||||
{
|
||||
parent::reset();
|
||||
|
||||
/*
|
||||
* When \Behat\Mink\Driver\DriverInterface::visit() is called on this driver here,
|
||||
* we ultimately end up in \Symfony\Bundle\FrameworkBundle\KernelBrowser::doRequest().
|
||||
* That method tracks state across multiple requests to detect whether it is necessary
|
||||
* to reboot the targeted-at kernel before performing the next request.
|
||||
*
|
||||
* We do not want this state to leak between Behat scenarios, and so this method here
|
||||
* seems to be a good place to reset driver state as well.
|
||||
*
|
||||
* Since there is no other way to reset the KernelBrowser, we create a new instance.
|
||||
*
|
||||
* This also makes sense for another reason: The $kernel instance is rebooted by the
|
||||
* KernelOrchestrator between Behat scenarios. So, every time we reset the driver
|
||||
* (which happens at least for the first request during a scenario) we want to make
|
||||
* sure we are using a KernelBrowser instance created in the currently active
|
||||
* kernel "state" ("epoch"? "generation"?)
|
||||
*/
|
||||
|
||||
parent::__construct($this->createBrowser(), $this->baseUrl);
|
||||
}
|
||||
|
||||
private function createBrowser(): AbstractBrowser
|
||||
{
|
||||
/** @var object $testClient */
|
||||
$testClient = $this->kernel->getContainer()->get('test.client');
|
||||
|
||||
if (!$testClient instanceof AbstractBrowser) {
|
||||
throw new \RuntimeException(sprintf(
|
||||
'Service "test.client" should be an instance of "%s" or "%s", "%s" given.',
|
||||
Client::class,
|
||||
'Service "test.client" should be an instance of "%s", "%s" given.',
|
||||
AbstractBrowser::class,
|
||||
get_class($testClient),
|
||||
));
|
||||
}
|
||||
|
||||
parent::__construct($testClient, $baseUrl);
|
||||
return $testClient;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,12 +15,16 @@ final class KernelOrchestrator implements EventSubscriberInterface
|
||||
/** @var KernelInterface */
|
||||
private $symfonyKernel;
|
||||
|
||||
/** @var KernelInterface */
|
||||
private $driverKernel;
|
||||
|
||||
/** @var ContainerInterface */
|
||||
private $behatContainer;
|
||||
|
||||
public function __construct(KernelInterface $symfonyKernel, ContainerInterface $behatContainer)
|
||||
public function __construct(KernelInterface $symfonyKernel, KernelInterface $driverKernel, ContainerInterface $behatContainer)
|
||||
{
|
||||
$this->symfonyKernel = $symfonyKernel;
|
||||
$this->driverKernel = $driverKernel;
|
||||
$this->behatContainer = $behatContainer;
|
||||
}
|
||||
|
||||
@@ -42,8 +46,25 @@ final class KernelOrchestrator implements EventSubscriberInterface
|
||||
|
||||
public function tearDown(): void
|
||||
{
|
||||
$this->driverKernel->shutdown();
|
||||
|
||||
/*
|
||||
* Reset both Kernel instances after a scenario has been run: The Kernel (and thus Container)
|
||||
* used in Behat to configure Contexts; and the Kernel used by the SymfonyDriver to which
|
||||
* requests are dispatched (through Mink).
|
||||
*
|
||||
* Since the "symfony" container is needed in a few other places (where and why exactly?) and
|
||||
* has to be in a booted/usable state most of the time, we do not shut it down here in tearDown()
|
||||
* and boot it in setUp().
|
||||
*
|
||||
* Instead, the definitions in \FriendsOfBehat\SymfonyExtension\ServiceContainer\SymfonyExtension
|
||||
* make sure both kernels are booted immediately after being created, and we also initiate the
|
||||
* re-boot() here right away.
|
||||
*/
|
||||
$this->symfonyKernel->getContainer()->set('behat.service_container', null);
|
||||
$this->symfonyKernel->shutdown();
|
||||
$this->symfonyKernel->boot();
|
||||
|
||||
$this->driverKernel->boot();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -134,7 +134,7 @@ final class SymfonyExtension implements Extension
|
||||
|
||||
private function loadKernelRebooter(ContainerBuilder $container): void
|
||||
{
|
||||
$definition = new Definition(KernelOrchestrator::class, [new Reference(self::KERNEL_ID), $container]);
|
||||
$definition = new Definition(KernelOrchestrator::class, [new Reference(self::KERNEL_ID), new Reference(self::DRIVER_KERNEL_ID), $container]);
|
||||
$definition->addTag(EventDispatcherExtension::SUBSCRIBER_TAG);
|
||||
|
||||
$container->setDefinition('fob_symfony.kernel_orchestrator', $definition);
|
||||
|
||||
@@ -167,7 +167,7 @@ final class Controller
|
||||
{
|
||||
$this->counter->increase();
|
||||
|
||||
return new Response('Hello world!');
|
||||
return new Response('Hello world! The counter value is ' . $this->counter->get());
|
||||
}
|
||||
}
|
||||
CON
|
||||
|
||||
Reference in New Issue
Block a user