From 09aadef83b9ec5c8f2e507f62c1eb6e9e6aea2c8 Mon Sep 17 00:00:00 2001 From: Christophe Coevoet Date: Fri, 29 Jun 2018 13:01:08 +0200 Subject: [PATCH] Fix support for URL to account for master-slave and pooling-shard connections --- lib/Doctrine/DBAL/DriverManager.php | 22 ++++++ .../Doctrine/Tests/DBAL/DriverManagerTest.php | 71 +++++++++++++++++++ 2 files changed, 93 insertions(+) diff --git a/lib/Doctrine/DBAL/DriverManager.php b/lib/Doctrine/DBAL/DriverManager.php index 22ae8ac3264..6f7164e9e0e 100644 --- a/lib/Doctrine/DBAL/DriverManager.php +++ b/lib/Doctrine/DBAL/DriverManager.php @@ -155,6 +155,28 @@ public static function getConnection( $params = self::parseDatabaseUrl($params); + // URL support for MasterSlaveConnection + if (isset($params['master'])) { + $params['master'] = self::parseDatabaseUrl($params['master']); + } + + if (isset($params['slaves'])) { + foreach ($params['slaves'] as $key => $slaveParams) { + $params['slaves'][$key] = self::parseDatabaseUrl($slaveParams); + } + } + + // URL support for PoolingShardConnection + if (isset($params['global'])) { + $params['global'] = self::parseDatabaseUrl($params['global']); + } + + if (isset($params['shards'])) { + foreach ($params['shards'] as $key => $shardParams) { + $params['shards'][$key] = self::parseDatabaseUrl($shardParams); + } + } + // check for existing pdo object if (isset($params['pdo']) && ! $params['pdo'] instanceof \PDO) { throw DBALException::invalidPdoInstance(); diff --git a/tests/Doctrine/Tests/DBAL/DriverManagerTest.php b/tests/Doctrine/Tests/DBAL/DriverManagerTest.php index 05892a2d0ff..7e7c5b0153a 100644 --- a/tests/Doctrine/Tests/DBAL/DriverManagerTest.php +++ b/tests/Doctrine/Tests/DBAL/DriverManagerTest.php @@ -2,12 +2,15 @@ namespace Doctrine\Tests\DBAL; +use Doctrine\DBAL\Connections\MasterSlaveConnection; use Doctrine\DBAL\DBALException; use Doctrine\DBAL\Driver\DrizzlePDOMySql\Driver as DrizzlePDOMySqlDriver; use Doctrine\DBAL\Driver\PDOMySql\Driver as PDOMySQLDriver; use Doctrine\DBAL\Driver\PDOSqlite\Driver as PDOSqliteDriver; use Doctrine\DBAL\Driver\SQLSrv\Driver as SQLSrvDriver; use Doctrine\DBAL\DriverManager; +use Doctrine\DBAL\Sharding\PoolingShardConnection; +use Doctrine\DBAL\Sharding\ShardChoser\MultiTenantShardChoser; use Doctrine\Tests\DBAL\Mocks\MockPlatform; use Doctrine\Tests\DbalTestCase; use Doctrine\Tests\Mocks\ConnectionMock; @@ -133,6 +136,74 @@ public function testValidDriverClass() self::assertInstanceOf(PDOMySQLDriver::class, $conn->getDriver()); } + public function testDatabaseUrlMasterSlave() + { + $options = [ + 'driver' => 'pdo_mysql', + 'master' => ['url' => 'mysql://foo:bar@localhost:11211/baz'], + 'slaves' => [ + 'slave1' => ['url' => 'mysql://foo:bar@localhost:11211/baz_slave'], + ], + 'wrapperClass' => MasterSlaveConnection::class, + ]; + + $conn = DriverManager::getConnection($options); + + $params = $conn->getParams(); + self::assertInstanceOf(PDOMySQLDriver::class, $conn->getDriver()); + + $expected = [ + 'user' => 'foo', + 'password' => 'bar', + 'host' => 'localhost', + 'port' => 11211, + ]; + + foreach ($expected as $key => $value) { + self::assertEquals($value, $params['master'][$key]); + self::assertEquals($value, $params['slaves']['slave1'][$key]); + } + + self::assertEquals('baz', $params['master']['dbname']); + self::assertEquals('baz_slave', $params['slaves']['slave1']['dbname']); + } + + public function testDatabaseUrlShard() + { + $options = [ + 'driver' => 'pdo_mysql', + 'shardChoser' => MultiTenantShardChoser::class, + 'global' => ['url' => 'mysql://foo:bar@localhost:11211/baz'], + 'shards' => [ + [ + 'id' => 1, + 'url' => 'mysql://foo:bar@localhost:11211/baz_slave', + ], + ], + 'wrapperClass' => PoolingShardConnection::class, + ]; + + $conn = DriverManager::getConnection($options); + + $params = $conn->getParams(); + self::assertInstanceOf(PDOMySQLDriver::class, $conn->getDriver()); + + $expected = [ + 'user' => 'foo', + 'password' => 'bar', + 'host' => 'localhost', + 'port' => 11211, + ]; + + foreach ($expected as $key => $value) { + self::assertEquals($value, $params['global'][$key]); + self::assertEquals($value, $params['shards'][0][$key]); + } + + self::assertEquals('baz', $params['global']['dbname']); + self::assertEquals('baz_slave', $params['shards'][0]['dbname']); + } + /** * @dataProvider databaseUrls */