'ibm_db2', 'mssql' => 'pdo_sqlsrv', 'mysql' => 'pdo_mysql', 'mysql2' => 'pdo_mysql', // Amazon RDS, for some weird reason 'postgres' => 'pdo_pgsql', 'postgresql' => 'pdo_pgsql', 'pgsql' => 'pdo_pgsql', 'sqlite' => 'pdo_sqlite', 'sqlite3' => 'pdo_sqlite', ]; /** @var mixed[][] */ private array $typesConfig = []; private DsnParser $dsnParser; private bool $initialized = false; /** @param mixed[][] $typesConfig */ public function __construct(array $typesConfig, ?DsnParser $dsnParser = null) { $this->typesConfig = $typesConfig; $this->dsnParser = $dsnParser ?? new DsnParser(self::DEFAULT_SCHEME_MAP); } /** * Create a connection by name. * * @param mixed[] $params * @param array $mappingTypes * @psalm-param Params $params * * @return Connection */ public function createConnection(array $params, ?Configuration $config = null, ?EventManager $eventManager = null, array $mappingTypes = []) { if (! $this->initialized) { $this->initializeTypes(); } $overriddenOptions = []; if (isset($params['connection_override_options'])) { trigger_deprecation('doctrine/doctrine-bundle', '2.4', 'The "connection_override_options" connection parameter is deprecated'); $overriddenOptions = $params['connection_override_options']; unset($params['connection_override_options']); } $params = $this->parseDatabaseUrl($params); // URL support for PrimaryReplicaConnection if (isset($params['primary'])) { $params['primary'] = $this->parseDatabaseUrl($params['primary']); } if (isset($params['replica'])) { foreach ($params['replica'] as $key => $replicaParams) { $params['replica'][$key] = $this->parseDatabaseUrl($replicaParams); } } if (! isset($params['pdo']) && (! isset($params['charset']) || $overriddenOptions || isset($params['dbname_suffix']))) { $wrapperClass = null; if (isset($params['wrapperClass'])) { if (! is_subclass_of($params['wrapperClass'], Connection::class)) { throw DBALException::invalidWrapperClass($params['wrapperClass']); } $wrapperClass = $params['wrapperClass']; $params['wrapperClass'] = null; } $connection = DriverManager::getConnection($params, $config, $eventManager); $params = $this->addDatabaseSuffix(array_merge($connection->getParams(), $overriddenOptions)); $driver = $connection->getDriver(); $platform = $driver->getDatabasePlatform(); if (! isset($params['charset'])) { if ($platform instanceof AbstractMySQLPlatform) { $params['charset'] = 'utf8mb4'; if (isset($params['defaultTableOptions']['collate'])) { Deprecation::trigger( 'doctrine/doctrine-bundle', 'https://github.com/doctrine/dbal/issues/5214', 'The "collate" default table option is deprecated in favor of "collation" and will be removed in doctrine/doctrine-bundle 3.0. ' ); $params['defaultTableOptions']['collation'] = $params['defaultTableOptions']['collate']; unset($params['defaultTableOptions']['collate']); } if (! isset($params['defaultTableOptions']['collation'])) { $params['defaultTableOptions']['collation'] = 'utf8mb4_unicode_ci'; } } else { $params['charset'] = 'utf8'; } } if ($wrapperClass !== null) { $params['wrapperClass'] = $wrapperClass; } else { $wrapperClass = Connection::class; } $connection = new $wrapperClass($params, $driver, $config, $eventManager); } else { $connection = DriverManager::getConnection($params, $config, $eventManager); } if (! empty($mappingTypes)) { $platform = $this->getDatabasePlatform($connection); foreach ($mappingTypes as $dbType => $doctrineType) { $platform->registerDoctrineTypeMapping($dbType, $doctrineType); } } return $connection; } /** * Try to get the database platform. * * This could fail if types should be registered to an predefined/unused connection * and the platform version is unknown. * * @link https://github.com/doctrine/DoctrineBundle/issues/673 * * @throws DBALException */ private function getDatabasePlatform(Connection $connection): AbstractPlatform { try { return $connection->getDatabasePlatform(); } catch (DriverException $driverException) { throw new DBALException( 'An exception occurred while establishing a connection to figure out your platform version.' . PHP_EOL . "You can circumvent this by setting a 'server_version' configuration value" . PHP_EOL . PHP_EOL . 'For further information have a look at:' . PHP_EOL . 'https://github.com/doctrine/DoctrineBundle/issues/673', 0, $driverException ); } } /** * initialize the types */ private function initializeTypes(): void { foreach ($this->typesConfig as $typeName => $typeConfig) { if (Type::hasType($typeName)) { Type::overrideType($typeName, $typeConfig['class']); } else { Type::addType($typeName, $typeConfig['class']); } } $this->initialized = true; } /** * @param array $params * * @return array */ private function addDatabaseSuffix(array $params): array { if (isset($params['dbname']) && isset($params['dbname_suffix'])) { $params['dbname'] .= $params['dbname_suffix']; } foreach ($params['replica'] ?? [] as $key => $replicaParams) { if (! isset($replicaParams['dbname'], $replicaParams['dbname_suffix'])) { continue; } $params['replica'][$key]['dbname'] .= $replicaParams['dbname_suffix']; } if (isset($params['primary']['dbname'], $params['primary']['dbname_suffix'])) { $params['primary']['dbname'] .= $params['primary']['dbname_suffix']; } return $params; } /** * Extracts parts from a database URL, if present, and returns an * updated list of parameters. * * @param mixed[] $params The list of parameters. * @psalm-param Params $params * * @return mixed[] A modified list of parameters with info from a database * URL extracted into individual parameter parts. * @psalm-return Params * * @throws Exception */ private function parseDatabaseUrl(array $params): array { if (! isset($params['url'])) { return $params; } try { $parsedParams = $this->dsnParser->parse($params['url']); } catch (MalformedDsnException $e) { throw new Exception('Malformed parameter "url".', 0, $e); } if (isset($parsedParams['driver'])) { // The requested driver from the URL scheme takes precedence // over the default custom driver from the connection parameters (if any). unset($params['driverClass']); } $params = array_merge($params, $parsedParams); // If a schemeless connection URL is given, we require a default driver or default custom driver // as connection parameter. if (! isset($params['driverClass']) && ! isset($params['driver'])) { throw Exception::driverRequired($params['url']); } unset($params['url']); return $params; } }