Skip to content

Commit

Permalink
Merge pull request #1118 from ostrolucky/fix-inconsistent-maps-after-…
Browse files Browse the repository at this point in the history
…kernel-reset

Fix desynchronized entity managers after kernel reset
  • Loading branch information
alcaeus authored Jan 10, 2020
2 parents 08f9447 + 7c6f6ed commit 18fb7d2
Show file tree
Hide file tree
Showing 6 changed files with 122 additions and 64 deletions.
5 changes: 4 additions & 1 deletion Registry.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace Doctrine\Bundle\DoctrineBundle;

use Doctrine\ORM\EntityManager;
use Doctrine\ORM\EntityManagerInterface;
use Doctrine\ORM\ORMException;
use ProxyManager\Proxy\LazyLoadingInterface;
use Psr\Container\ContainerInterface;
Expand Down Expand Up @@ -182,7 +183,9 @@ private function resetOrClearManager(string $managerName, string $serviceId) : v

$manager = $this->container->get($serviceId);

if (! $manager instanceof LazyLoadingInterface) {
assert($manager instanceof EntityManagerInterface);

if (! $manager instanceof LazyLoadingInterface || $manager->isOpen()) {
$manager->clear();

return;
Expand Down
72 changes: 10 additions & 62 deletions Tests/Command/ImportMappingDoctrineCommandTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,32 +2,34 @@

namespace Doctrine\Bundle\DoctrineBundle\Tests\Command;

use Doctrine\Bundle\DoctrineBundle\DoctrineBundle;
use Doctrine\Bundle\DoctrineBundle\Tests\DependencyInjection\Fixtures\TestKernel;
use PHPUnit\Framework\TestCase;
use Psr\Log\NullLogger;
use Symfony\Bundle\FrameworkBundle\Console\Application;
use Symfony\Bundle\FrameworkBundle\FrameworkBundle;
use Symfony\Component\Config\Loader\LoaderInterface;
use Symfony\Component\Console\Tester\CommandTester;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\Filesystem\Filesystem;
use Symfony\Component\HttpKernel\Bundle\Bundle;
use Symfony\Component\HttpKernel\Kernel;

/**
* @group legacy
*/
class ImportMappingDoctrineCommandTest extends TestCase
{
/** @var ImportMappingTestingKernel|null */
/** @var TestKernel|null */
private $kernel;

/** @var CommandTester|null */
private $commandTester;

protected function setup()
{
$this->kernel = new ImportMappingTestingKernel();
$this->kernel = new class() extends TestKernel {
public function registerBundles() : iterable
{
yield from parent::registerBundles();
yield new ImportMappingTestFooBundle();
}
};

$this->kernel->boot();

$connection = $this->kernel->getContainer()
Expand Down Expand Up @@ -108,60 +110,6 @@ public function testExecuteAnnotationsWithNamespace()
}
}

class ImportMappingTestingKernel extends Kernel
{
/** @var string|null */
private $projectDir;

public function __construct()
{
parent::__construct('test', true);
}

public function registerBundles() : iterable
{
return [
new FrameworkBundle(),
new DoctrineBundle(),
new ImportMappingTestFooBundle(),
];
}

public function registerContainerConfiguration(LoaderInterface $loader)
{
$loader->load(function (ContainerBuilder $container) {
// @todo Setting the kernel.name parameter can be removed once the dependency on DoctrineCacheBundle has been dropped
$container->setParameter('kernel.name', 'foo');
$container->loadFromExtension('framework', ['secret' => 'F00']);

$container->loadFromExtension('doctrine', [
'dbal' => [
'driver' => 'pdo_sqlite',
'path' => $this->getCacheDir() . '/testing.db',
],
'orm' => [],
]);

// Register a NullLogger to avoid getting the stderr default logger of FrameworkBundle
$container->register('logger', NullLogger::class);
});
}

public function getProjectDir() : string
{
if ($this->projectDir === null) {
$this->projectDir = sys_get_temp_dir() . '/sf_kernel_' . md5(mt_rand());
}

return $this->projectDir;
}

public function getRootDir() : string
{
return $this->getProjectDir();
}
}

class ImportMappingTestFooBundle extends Bundle
{
public function getPath() : string
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,13 @@

namespace Fixtures\Bundles\RepositoryServiceBundle\Repository;

use Doctrine\ORM\EntityManager;
use Doctrine\ORM\EntityRepository;

class TestCustomClassRepoRepository extends EntityRepository
{
public function getEntityManager() : EntityManager
{
return parent::getEntityManager();
}
}
68 changes: 68 additions & 0 deletions Tests/DependencyInjection/Fixtures/TestKernel.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
<?php

namespace Doctrine\Bundle\DoctrineBundle\Tests\DependencyInjection\Fixtures;

use Doctrine\Bundle\DoctrineBundle\DoctrineBundle;
use Psr\Log\NullLogger;
use Symfony\Bundle\FrameworkBundle\FrameworkBundle;
use Symfony\Component\Config\Loader\LoaderInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\HttpKernel\Kernel;

class TestKernel extends Kernel
{
/** @var string|null */
private $projectDir;

public function __construct()
{
parent::__construct('test', true);
}

public function registerBundles() : iterable
{
return [
new FrameworkBundle(),
new DoctrineBundle(),
];
}

public function registerContainerConfiguration(LoaderInterface $loader)
{
$loader->load(static function (ContainerBuilder $container) {
// @todo Setting the kernel.name parameter can be removed once the dependency on DoctrineCacheBundle has been dropped
$container->setParameter('kernel.name', 'foo');
$container->loadFromExtension('framework', ['secret' => 'F00']);

$container->loadFromExtension('doctrine', [
'dbal' => ['driver' => 'pdo_sqlite'],
'orm' => [
'mappings' => [
'RepositoryServiceBundle' => [
'type' => 'annotation',
'dir' => __DIR__ . '/Bundles/RepositoryServiceBundle/Entity',
'prefix' => 'Fixtures\Bundles\RepositoryServiceBundle\Entity',
],
],
],
]);

// Register a NullLogger to avoid getting the stderr default logger of FrameworkBundle
$container->register('logger', NullLogger::class);
});
}

public function getProjectDir() : string
{
if ($this->projectDir === null) {
$this->projectDir = sys_get_temp_dir() . '/sf_kernel_' . md5(mt_rand());
}

return $this->projectDir;
}

public function getRootDir() : string
{
return $this->getProjectDir();
}
}
35 changes: 34 additions & 1 deletion Tests/RegistryTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,12 @@

use Closure;
use Doctrine\Bundle\DoctrineBundle\Registry;
use Doctrine\Bundle\DoctrineBundle\Tests\DependencyInjection\Fixtures\TestKernel;
use Doctrine\ORM\EntityManagerInterface;
use Fixtures\Bundles\RepositoryServiceBundle\Entity\TestCustomClassRepoEntity;
use Fixtures\Bundles\RepositoryServiceBundle\Repository\TestCustomClassRepoRepository;
use ProxyManager\Proxy\LazyLoadingInterface;
use ProxyManager\Proxy\ProxyInterface;
use stdClass;

class RegistryTest extends TestCase
Expand Down Expand Up @@ -132,7 +136,7 @@ public function testReset()
$noProxyManager->expects($this->once())
->method('clear');

$proxyManager = $this->getMockBuilder(LazyLoadingInterface::class)->getMock();
$proxyManager = $this->getMockBuilder([LazyLoadingInterface::class, EntityManagerInterface::class])->getMock();
$proxyManager->expects($this->once())
->method('setProxyInitializer')
->with($this->isInstanceOf(Closure::class));
Expand All @@ -157,4 +161,33 @@ public function testReset()
$registry = new Registry($container, [], $entityManagers, 'default', 'default');
$registry->reset();
}

public function testIdentityMapsStayConsistentAfterReset()
{
$kernel = new TestKernel();
$kernel->boot();

$container = $kernel->getContainer();
$registry = $container->get('doctrine');
$entityManager = $container->get('doctrine.orm.default_entity_manager');
$repository = $entityManager->getRepository(TestCustomClassRepoEntity::class);

$this->assertInstanceOf(ProxyInterface::class, $entityManager);
assert($entityManager instanceof EntityManagerInterface);
assert($registry instanceof Registry);
assert($repository instanceof TestCustomClassRepoRepository);

$entity = new TestCustomClassRepoEntity();
$repository->getEntityManager()->persist($entity);

$this->assertTrue($entityManager->getUnitOfWork()->isEntityScheduled($entity));
$this->assertTrue($repository->getEntityManager()->getUnitOfWork()->isEntityScheduled($entity));

$registry->reset();

$this->assertFalse($entityManager->getUnitOfWork()->isEntityScheduled($entity));
$this->assertFalse($repository->getEntityManager()->getUnitOfWork()->isEntityScheduled($entity));

$entityManager->flush();
}
}
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
"phpunit/phpunit": "^7.5",
"symfony/phpunit-bridge": "^4.2",
"symfony/property-info": "^3.4.30|^4.3.3",
"symfony/proxy-manager-bridge": "^3.4|^4|^5",
"symfony/twig-bridge": "^3.4|^4.1",
"symfony/validator": "^3.4.30|^4.3.3",
"symfony/web-profiler-bundle": "^3.4.30|^4.3.3",
Expand Down

0 comments on commit 18fb7d2

Please sign in to comment.