diff --git a/config/dbal.xml b/config/dbal.xml index a38d0de9..61fcb5ed 100644 --- a/config/dbal.xml +++ b/config/dbal.xml @@ -101,6 +101,12 @@ + + + + + + diff --git a/config/middlewares.xml b/config/middlewares.xml index d6bf92bc..00dad9c4 100644 --- a/config/middlewares.xml +++ b/config/middlewares.xml @@ -5,6 +5,7 @@ xsi:schemaLocation="http://symfony.com/schema/dic/services http://symfony.com/schema/dic/services/services-1.0.xsd"> + @@ -17,5 +18,9 @@ + + + + diff --git a/psalm.xml.dist b/psalm.xml.dist index 9a319d21..16e9f0db 100644 --- a/psalm.xml.dist +++ b/psalm.xml.dist @@ -43,6 +43,8 @@ + + diff --git a/src/DependencyInjection/Configuration.php b/src/DependencyInjection/Configuration.php index ba0fed1d..65c680df 100644 --- a/src/DependencyInjection/Configuration.php +++ b/src/DependencyInjection/Configuration.php @@ -221,6 +221,7 @@ private function getDbalConnectionsNode(): ArrayNodeDefinition ->end() ->booleanNode('disable_type_comments')->end() ->scalarNode('server_version')->end() + ->integerNode('idle_connection_ttl')->defaultValue(600)->end() ->scalarNode('driver_class')->end() ->scalarNode('wrapper_class')->end() ->booleanNode('keep_slave') diff --git a/src/DependencyInjection/DoctrineExtension.php b/src/DependencyInjection/DoctrineExtension.php index 161626a7..4dfe3f2c 100644 --- a/src/DependencyInjection/DoctrineExtension.php +++ b/src/DependencyInjection/DoctrineExtension.php @@ -37,6 +37,7 @@ use Symfony\Bridge\Doctrine\DependencyInjection\AbstractDoctrineExtension; use Symfony\Bridge\Doctrine\IdGenerator\UlidGenerator; use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator; +use Symfony\Bridge\Doctrine\Middleware\IdleConnection\Listener; use Symfony\Bridge\Doctrine\PropertyInfo\DoctrineExtractor; use Symfony\Bridge\Doctrine\SchemaListener\DoctrineDbalCacheAdapterSchemaListener; use Symfony\Bridge\Doctrine\SchemaListener\LockStoreSchemaListener; @@ -83,7 +84,7 @@ * * @final since 2.9 * @psalm-type DBALConfig = array{ - * connections: array, + * connections: array, * driver_schemes: array, * default_connection: string, * types: array, @@ -196,6 +197,8 @@ protected function dbalLoad(array $config, ContainerBuilder $container) $connWithLogging = []; $connWithProfiling = []; $connWithBacktrace = []; + $ttlByConnection = []; + foreach ($config['connections'] as $name => $connection) { if ($connection['logging']) { $connWithLogging[] = $name; @@ -209,6 +212,10 @@ protected function dbalLoad(array $config, ContainerBuilder $container) } } + if ($connection['idle_connection_ttl'] > 0) { + $ttlByConnection[$name] = $connection['idle_connection_ttl']; + } + $this->loadDbalConnection($name, $connection, $container); } @@ -228,7 +235,16 @@ protected function dbalLoad(array $config, ContainerBuilder $container) } }); - $this->registerDbalMiddlewares($container, $connWithLogging, $connWithProfiling, $connWithBacktrace); + $this->registerDbalMiddlewares($container, $connWithLogging, $connWithProfiling, $connWithBacktrace, array_keys($ttlByConnection)); + + $container->getDefinition('doctrine.dbal.idle_connection_middleware')->setArgument(1, $ttlByConnection); + + if (class_exists(Listener::class)) { + return; + } + + $container->removeDefinition('doctrine.dbal.idle_connection_listener'); + $container->removeDefinition('doctrine.dbal.idle_connection_middleware'); } /** @@ -1186,12 +1202,14 @@ private function createArrayAdapterCachePool(ContainerBuilder $container, string * @param string[] $connWithLogging * @param string[] $connWithProfiling * @param string[] $connWithBacktrace + * @param string[] $connWithTtl */ private function registerDbalMiddlewares( ContainerBuilder $container, array $connWithLogging, array $connWithProfiling, - array $connWithBacktrace + array $connWithBacktrace, + array $connWithTtl ): void { $loader = new XmlFileLoader($container, new FileLocator(__DIR__ . '/../../config')); $loader->load('middlewares.xml'); @@ -1207,5 +1225,11 @@ private function registerDbalMiddlewares( $debugMiddlewareAbstractDef ->addTag('doctrine.middleware', ['connection' => $connName, 'priority' => 10]); } + + $idleConnectionMiddlewareAbstractDef = $container->getDefinition('doctrine.dbal.idle_connection_middleware'); + foreach ($connWithTtl as $connName) { + $idleConnectionMiddlewareAbstractDef + ->addTag('doctrine.middleware', ['connection' => $connName, 'priority' => 10]); + } } } diff --git a/src/Middleware/IdleConnectionMiddleware.php b/src/Middleware/IdleConnectionMiddleware.php new file mode 100644 index 00000000..d64d22fa --- /dev/null +++ b/src/Middleware/IdleConnectionMiddleware.php @@ -0,0 +1,36 @@ + */ + private array $ttlByConnection; + private string $connectionName; + + /** + * @param ArrayObject $connectionExpiries + * @param array $ttlByConnection + */ + public function __construct(ArrayObject $connectionExpiries, array $ttlByConnection) + { + $this->connectionExpiries = $connectionExpiries; + $this->ttlByConnection = $ttlByConnection; + } + + public function setConnectionName(string $name): void + { + $this->connectionName = $name; + } + + public function wrap(Driver $driver): IdleConnectionDriver + { + return new IdleConnectionDriver($driver, $this->connectionExpiries, $this->ttlByConnection[$this->connectionName], $this->connectionName); + } +} diff --git a/tests/DependencyInjection/AbstractDoctrineExtensionTest.php b/tests/DependencyInjection/AbstractDoctrineExtensionTest.php index 58c0fcf3..60daad64 100644 --- a/tests/DependencyInjection/AbstractDoctrineExtensionTest.php +++ b/tests/DependencyInjection/AbstractDoctrineExtensionTest.php @@ -220,6 +220,7 @@ public function testDbalLoadSinglePrimaryReplicaConnection(): void 'host' => 'localhost', 'unix_socket' => '/path/to/mysqld.sock', 'driverOptions' => [PDO::ATTR_STRINGIFY_FETCHES => 1], + 'idle_connection_ttl' => 600, ], $param['primary'], ); @@ -340,6 +341,7 @@ public function testLoadSimpleSingleConnection(): void 'driver' => 'pdo_mysql', 'driverOptions' => [], 'defaultTableOptions' => [], + 'idle_connection_ttl' => 600, ], new Reference('doctrine.dbal.default_connection.configuration'), method_exists(Connection::class, 'getEventManager') @@ -379,6 +381,7 @@ public function testLoadSimpleSingleConnectionWithoutDbName(): void 'driver' => 'pdo_mysql', 'driverOptions' => [], 'defaultTableOptions' => [], + 'idle_connection_ttl' => 600, ], new Reference('doctrine.dbal.default_connection.configuration'), method_exists(Connection::class, 'getEventManager') @@ -418,6 +421,7 @@ public function testLoadSingleConnection(): void 'dbname' => 'sqlite_db', 'memory' => true, 'defaultTableOptions' => [], + 'idle_connection_ttl' => 600, ], new Reference('doctrine.dbal.default_connection.configuration'), method_exists(Connection::class, 'getEventManager') diff --git a/tests/DependencyInjection/Compiler/MiddlewarePassTest.php b/tests/DependencyInjection/Compiler/MiddlewarePassTest.php index 06dc92cb..acb64f93 100644 --- a/tests/DependencyInjection/Compiler/MiddlewarePassTest.php +++ b/tests/DependencyInjection/Compiler/MiddlewarePassTest.php @@ -6,15 +6,18 @@ use Doctrine\Bundle\DoctrineBundle\DependencyInjection\Compiler\MiddlewaresPass; use Doctrine\Bundle\DoctrineBundle\DependencyInjection\DoctrineExtension; use Doctrine\Bundle\DoctrineBundle\Middleware\ConnectionNameAwareInterface; +use Doctrine\Bundle\DoctrineBundle\Middleware\IdleConnectionMiddleware; use Doctrine\DBAL\Driver; use Doctrine\DBAL\Driver\Middleware; use PHPUnit\Framework\TestCase; use Psr\Log\NullLogger; +use Symfony\Bridge\Doctrine\Middleware\IdleConnection\Listener; use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\Definition; use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag; use function array_map; +use function class_exists; use function implode; use function sprintf; @@ -170,7 +173,8 @@ public function testAddMiddlewareOrderingWithDefaultPriority(): void $this->assertMiddlewareInjected($container, 'conn1', PHP7Middleware::class); $this->assertMiddlewareInjected($container, 'conn1', ConnectionAwarePHP7Middleware::class, true); - $this->assertMiddlewareOrdering($container, 'conn1', [PHP7Middleware::class, ConnectionAwarePHP7Middleware::class]); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, PHP7Middleware::class, ConnectionAwarePHP7Middleware::class] : [PHP7Middleware::class, ConnectionAwarePHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); } public function testAddMiddlewareOrderingWithExplicitPriority(): void @@ -193,7 +197,8 @@ public function testAddMiddlewareOrderingWithExplicitPriority(): void $this->assertMiddlewareInjected($container, 'conn1', PHP7Middleware::class); $this->assertMiddlewareInjected($container, 'conn1', ConnectionAwarePHP7Middleware::class, true); - $this->assertMiddlewareOrdering($container, 'conn1', [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, ConnectionAwarePHP7Middleware::class, PHP7Middleware::class] : [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); } public function testAddMiddlewareOrderingWithExplicitPriorityAndConnection(): void @@ -222,7 +227,8 @@ public function testAddMiddlewareOrderingWithExplicitPriorityAndConnection(): vo $this->assertMiddlewareInjected($container, 'conn1', ConnectionAwarePHP7Middleware::class, true); $this->assertMiddlewareInjected($container, 'conn2', PHP7Middleware::class); $this->assertMiddlewareNotInjected($container, 'conn2', ConnectionAwarePHP7Middleware::class); - $this->assertMiddlewareOrdering($container, 'conn1', [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, ConnectionAwarePHP7Middleware::class, PHP7Middleware::class] : [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); } public function testAddMiddlewareOrderingWithExplicitPriorityPerConnection(): void @@ -252,8 +258,10 @@ public function testAddMiddlewareOrderingWithExplicitPriorityPerConnection(): vo $this->assertMiddlewareInjected($container, 'conn1', ConnectionAwarePHP7Middleware::class, true); $this->assertMiddlewareInjected($container, 'conn2', PHP7Middleware::class); $this->assertMiddlewareInjected($container, 'conn2', ConnectionAwarePHP7Middleware::class, true); - $this->assertMiddlewareOrdering($container, 'conn1', [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]); - $this->assertMiddlewareOrdering($container, 'conn2', [PHP7Middleware::class, ConnectionAwarePHP7Middleware::class]); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, ConnectionAwarePHP7Middleware::class, PHP7Middleware::class] : [ConnectionAwarePHP7Middleware::class, PHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, PHP7Middleware::class, ConnectionAwarePHP7Middleware::class] : [PHP7Middleware::class, ConnectionAwarePHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn2', $expectedMiddlewares); } public function testAddMiddlewareOrderingWithInheritedPriorityPerConnection(): void @@ -292,8 +300,10 @@ public function testAddMiddlewareOrderingWithInheritedPriorityPerConnection(): v $this->assertMiddlewareInjected($container, 'conn2', PHP7Middleware::class); $this->assertMiddlewareNotInjected($container, 'conn2', ConnectionAwarePHP7Middleware::class); $this->assertMiddlewareInjected($container, 'conn2', 'some_middleware_class'); - $this->assertMiddlewareOrdering($container, 'conn1', [ConnectionAwarePHP7Middleware::class, 'some_middleware_class', PHP7Middleware::class]); - $this->assertMiddlewareOrdering($container, 'conn2', [PHP7Middleware::class, 'some_middleware_class']); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, ConnectionAwarePHP7Middleware::class, 'some_middleware_class', PHP7Middleware::class] : [ConnectionAwarePHP7Middleware::class, 'some_middleware_class', PHP7Middleware::class]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); + $expectedMiddlewares = class_exists(Listener::class) ? [IdleConnectionMiddleware::class, PHP7Middleware::class, 'some_middleware_class'] : [PHP7Middleware::class, 'some_middleware_class']; + $this->assertMiddlewareOrdering($container, 'conn2', $expectedMiddlewares); } /** @requires PHP 8 */ @@ -327,15 +337,28 @@ public function testAddMiddlewareOrderingWithAttributeForAutoconfiguration(): vo $this->assertMiddlewareInjected($container, 'conn2', AutoconfiguredMiddleware::class); $this->assertMiddlewareInjected($container, 'conn2', AutoconfiguredMiddlewareWithConnection::class); $this->assertMiddlewareInjected($container, 'conn2', AutoconfiguredMiddlewareWithPriority::class); - $this->assertMiddlewareOrdering($container, 'conn1', [ + $expectedMiddlewares = class_exists(Listener::class) ? [ + IdleConnectionMiddleware::class, AutoconfiguredMiddlewareWithPriority::class, AutoconfiguredMiddleware::class, - ]); - $this->assertMiddlewareOrdering($container, 'conn2', [ + ] : + [ + AutoconfiguredMiddlewareWithPriority::class, + AutoconfiguredMiddleware::class, + ]; + $this->assertMiddlewareOrdering($container, 'conn1', $expectedMiddlewares); + $expectedMiddlewares = class_exists(Listener::class) ? [ + IdleConnectionMiddleware::class, AutoconfiguredMiddlewareWithPriority::class, AutoconfiguredMiddleware::class, AutoconfiguredMiddlewareWithConnection::class, - ]); + ] : + [ + AutoconfiguredMiddlewareWithPriority::class, + AutoconfiguredMiddleware::class, + AutoconfiguredMiddlewareWithConnection::class, + ]; + $this->assertMiddlewareOrdering($container, 'conn2', $expectedMiddlewares); } private function createContainer(callable $func, bool $addConnections = true): ContainerBuilder diff --git a/tests/DependencyInjection/DoctrineExtensionTest.php b/tests/DependencyInjection/DoctrineExtensionTest.php index 1468a65d..904b5ad3 100644 --- a/tests/DependencyInjection/DoctrineExtensionTest.php +++ b/tests/DependencyInjection/DoctrineExtensionTest.php @@ -1424,6 +1424,57 @@ public function testDefinitionsToLogQueriesLoggingFalse(): void $this->assertArrayNotHasKey('doctrine.middleware', $abstractMiddlewareDefTags); } + /** @requires function Symfony\Bridge\Doctrine\Middleware\IdleConnection\Driver::__construct */ + public function testDefinitionsIdleConnection(): void + { + $container = $this->getContainer(); + $extension = new DoctrineExtension(); + + $config = BundleConfigurationBuilder::createBuilder() + ->addConnection([ + 'connections' => [ + 'conn1' => [ + 'password' => 'foo', + 'logging' => false, + 'profiling' => false, + 'idle_connection_ttl' => 15, + ], + 'conn2' => [ + 'password' => 'bar', + 'logging' => false, + 'profiling' => true, + ], + ], + ]) + ->build(); + + $extension->load([$config], $container); + + $this->assertTrue($container->hasDefinition('doctrine.dbal.idle_connection_middleware')); + + $abstractMiddlewareDef = $container->getDefinition('doctrine.dbal.idle_connection_middleware'); + $ttlByConnection = $abstractMiddlewareDef->getArgument(1); + + $this->assertArrayHasKey('conn1', $ttlByConnection); + $this->assertEquals(15, $ttlByConnection['conn1']); + $this->assertArrayHasKey('conn2', $ttlByConnection); + $this->assertEquals(600, $ttlByConnection['conn2']); + + $abstractMiddlewareDefTags = $container->getDefinition('doctrine.dbal.idle_connection_middleware')->getTags(); + + $idleConnectionMiddlewareTagAttributes = []; + foreach ($abstractMiddlewareDefTags as $tag => $attributes) { + if ($tag !== 'doctrine.middleware') { + continue; + } + + $idleConnectionMiddlewareTagAttributes = $attributes; + } + + $this->assertTrue(in_array(['connection' => 'conn1', 'priority' => 10], $idleConnectionMiddlewareTagAttributes, true), 'Tag with connection conn1 not found for doctrine.dbal.idle_connection_middleware'); + $this->assertTrue(in_array(['connection' => 'conn2', 'priority' => 10], $idleConnectionMiddlewareTagAttributes, true), 'Tag with connection conn2 found for doctrine.dbal.idle_connection_middleware'); + } + /** * @requires function \Symfony\Bridge\Doctrine\ArgumentResolver\EntityValueResolver::__construct * @testWith [true] diff --git a/tests/Middleware/IdleConnectionMiddlewareTest.php b/tests/Middleware/IdleConnectionMiddlewareTest.php new file mode 100644 index 00000000..40527a95 --- /dev/null +++ b/tests/Middleware/IdleConnectionMiddlewareTest.php @@ -0,0 +1,29 @@ + time() - 30, 'connectiontwo' => time() + 40]); + $ttlByConnection = ['connectionone' => 25, 'connectiontwo' => 60]; + + $middleware = new IdleConnectionMiddleware($connectionExpiries, $ttlByConnection); + $middleware->setConnectionName('connectionone'); + + $driverMock = $this->createMock(Driver::class); + $wrappedDriver = $middleware->wrap($driverMock); + + $this->assertInstanceOf(IdleConnectionDriver::class, $wrappedDriver); + } +}