Skip to content

Commit

Permalink
Add support for Messenger & Cache schema subscribers in Symfony 5.1
Browse files Browse the repository at this point in the history
  • Loading branch information
weaverryan committed May 8, 2020
1 parent 98e038d commit d7b61d6
Show file tree
Hide file tree
Showing 6 changed files with 138 additions and 0 deletions.
44 changes: 44 additions & 0 deletions DependencyInjection/Compiler/CacheSchemaSubscriberPass.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
<?php

namespace Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler;

use Symfony\Component\Cache\Adapter\PdoAdapter;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Reference;

/**
* Injects PdoAdapter into its schema subscriber.
*
* Must be run later after ResolveChildDefinitionsPass.
*/
class CacheSchemaSubscriberPass implements CompilerPassInterface
{
/**
* {@inheritDoc}
*/
public function process(ContainerBuilder $container)
{
$subscriberId = 'doctrine.orm.listeners.pdo_cache_adapter_doctrine_schema_subscriber';

if (! $container->hasDefinition($subscriberId)) {
return;
}

$cacheAdaptersReferences = [];
foreach ($container->getDefinitions() as $id => $definition) {
if ($definition->isAbstract() || $definition->isSynthetic()) {
continue;
}

if ($definition->getClass() !== PdoAdapter::class) {
continue;
}

$cacheAdaptersReferences[] = new Reference($id);
}

$container->getDefinition($subscriberId)
->replaceArgument(0, $cacheAdaptersReferences);
}
}
12 changes: 12 additions & 0 deletions DependencyInjection/DoctrineExtension.php
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@
use Symfony\Bridge\Doctrine\Messenger\DoctrineClearEntityManagerWorkerSubscriber;
use Symfony\Bridge\Doctrine\Messenger\DoctrineTransactionMiddleware;
use Symfony\Bridge\Doctrine\PropertyInfo\DoctrineExtractor;
use Symfony\Bridge\Doctrine\SchemaListener\MessengerTransportDoctrineSchemaSubscriber;
use Symfony\Bridge\Doctrine\SchemaListener\PdoCacheAdapterDoctrineSchemaSubscriber;
use Symfony\Bridge\Doctrine\Validator\DoctrineLoader;
use Symfony\Component\Cache\Adapter\ArrayAdapter;
use Symfony\Component\Cache\DoctrineProvider;
Expand Down Expand Up @@ -339,6 +341,11 @@ protected function ormLoad(array $config, ContainerBuilder $container)
$container->getDefinition('form.type.entity')->addTag('kernel.reset', ['method' => 'reset']);
}

// available in Symfony 5.1 and higher
if (! class_exists(PdoCacheAdapterDoctrineSchemaSubscriber::class)) {
$container->removeDefinition('doctrine.orm.listeners.pdo_cache_adapter_doctrine_schema_subscriber');
}

$entityManagers = [];
foreach (array_keys($config['entity_managers']) as $name) {
$entityManagers[$name] = sprintf('doctrine.orm.%s_entity_manager', $name);
Expand Down Expand Up @@ -842,6 +849,11 @@ private function loadMessengerServices(ContainerBuilder $container) : void
$container->removeDefinition('doctrine.orm.messenger.event_subscriber.doctrine_clear_entity_manager');
}

// available in Symfony 5.1 and higher
if (! class_exists(MessengerTransportDoctrineSchemaSubscriber::class)) {
$container->removeDefinition('doctrine.orm.messenger.doctrine_schema_subscriber');
}

$transportFactoryDefinition = $container->getDefinition('messenger.transport.doctrine.factory');
if (! class_exists(DoctrineTransportFactory::class)) {
// If symfony/messenger < 5.1
Expand Down
2 changes: 2 additions & 0 deletions DoctrineBundle.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

namespace Doctrine\Bundle\DoctrineBundle;

use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\CacheSchemaSubscriberPass;
use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\DbalSchemaFilterPass;
use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\EntityListenerPass;
use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\ServiceRepositoryCompilerPass;
Expand Down Expand Up @@ -40,6 +41,7 @@ public function build(ContainerBuilder $container)
$container->addCompilerPass(new ServiceRepositoryCompilerPass());
$container->addCompilerPass(new WellKnownSchemaFilterPass());
$container->addCompilerPass(new DbalSchemaFilterPass());
$container->addCompilerPass(new CacheSchemaSubscriberPass(), PassConfig::TYPE_OPTIMIZE, -10);
}

/**
Expand Down
5 changes: 5 additions & 0 deletions Resources/config/messenger.xml
Original file line number Diff line number Diff line change
Expand Up @@ -40,5 +40,10 @@
<tag name="kernel.event_subscriber" />
<argument type="service" id="doctrine" />
</service>

<service id="doctrine.orm.messenger.doctrine_schema_subscriber" class="Symfony\Bridge\Doctrine\SchemaListener\MessengerTransportDoctrineSchemaSubscriber">
<argument type="tagged" tag="messenger.receiver" />
<tag name="doctrine.event_subscriber" />
</service>
</services>
</container>
4 changes: 4 additions & 0 deletions Resources/config/orm.xml
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,10 @@

<!-- listeners -->
<service id="doctrine.orm.listeners.resolve_target_entity" class="%doctrine.orm.listeners.resolve_target_entity.class%" public="false" />
<service id="doctrine.orm.listeners.pdo_cache_adapter_doctrine_schema_subscriber" class="Symfony\Bridge\Doctrine\SchemaListener\PdoCacheAdapterDoctrineSchemaSubscriber">
<argument type="collection" /> <!-- PdoAdapter instances -->
<tag name="doctrine.event_subscriber" />
</service>

<!-- naming strategy -->
<service id="doctrine.orm.naming_strategy.default" class="%doctrine.orm.naming_strategy.default.class%" public="false" />
Expand Down
71 changes: 71 additions & 0 deletions Tests/CacheSchemaSubscriberTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
<?php

namespace Doctrine\Bundle\DoctrineBundle\Tests;

use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\CacheSchemaSubscriberPass;
use Doctrine\Bundle\DoctrineBundle\DependencyInjection\DoctrineExtension;
use Symfony\Bridge\Doctrine\SchemaListener\PdoCacheAdapterDoctrineSchemaSubscriber;
use Symfony\Bundle\FrameworkBundle\DependencyInjection\FrameworkExtension;
use Symfony\Component\DependencyInjection\Alias;
use Symfony\Component\DependencyInjection\Compiler\PassConfig;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag;
use Symfony\Component\DependencyInjection\Reference;

class CacheSchemaSubscriberTest extends TestCase
{
public function testSchemaSubscriberWiring() : void
{
if (! class_exists(PdoCacheAdapterDoctrineSchemaSubscriber::class)) {
$this->markTestSkipped('This test requires Symfony 5.1 or higher');
}

$container = new ContainerBuilder(new ParameterBag([
'kernel.name' => 'app',
'kernel.debug' => false,
'kernel.bundles' => [],
'kernel.cache_dir' => sys_get_temp_dir(),
'kernel.environment' => 'test',
'kernel.root_dir' => __DIR__ . '/../../../../', // src dir
'kernel.project_dir' => __DIR__ . '/../../../../', // src dir
'kernel.bundles_metadata' => [],
'kernel.charset' => 'UTF-8',
'kernel.container_class' => ContainerBuilder::class,
'kernel.secret' => 'test',
'env(base64:default::SYMFONY_DECRYPTION_SECRET)' => 'foo',
]));

$extension = new FrameworkExtension();
$container->registerExtension($extension);
$extension->load([
'framework' => [
'cache' => [
'pools' => [
'my_cache_adapter' => ['adapter' => 'cache.adapter.pdo'],
],
],
],
], $container);

$extension = new DoctrineExtension();
$container->registerExtension($extension);
$extension->load([
[
'dbal' => [],
'orm' => [],
],
], $container);

$container->setAlias('test_subscriber_alias', new Alias('doctrine.orm.listeners.pdo_cache_adapter_doctrine_schema_subscriber', true));
// prevent my_cache_apapter from inlining
$container->register('uses_my_cache_adapter', 'stdClass')
->addArgument(new Reference('my_cache_adapter'))
->setPublic(true);
$container->addCompilerPass(new CacheSchemaSubscriberPass(), PassConfig::TYPE_OPTIMIZE, -10);
$container->compile();

// check that PdoAdapter service is injected as an argument
$definition = $container->findDefinition('test_subscriber_alias');
$this->assertEquals([new Reference('my_cache_adapter')], $definition->getArgument(0));
}
}

0 comments on commit d7b61d6

Please sign in to comment.