DriverManager.php 14.7 KB
Newer Older
romanb's avatar
romanb committed
1 2
<?php

3 4 5
namespace Doctrine\DBAL;

use Doctrine\Common\EventManager;
6 7 8 9 10 11 12 13 14 15 16
use Doctrine\DBAL\Driver\DrizzlePDOMySql\Driver as DrizzlePDOMySQLDriver;
use Doctrine\DBAL\Driver\IBMDB2\DB2Driver;
use Doctrine\DBAL\Driver\Mysqli\Driver as MySQLiDriver;
use Doctrine\DBAL\Driver\OCI8\Driver as OCI8Driver;
use Doctrine\DBAL\Driver\PDOMySql\Driver as PDOMySQLDriver;
use Doctrine\DBAL\Driver\PDOOracle\Driver as PDOOCIDriver;
use Doctrine\DBAL\Driver\PDOPgSql\Driver as PDOPgSQLDriver;
use Doctrine\DBAL\Driver\PDOSqlite\Driver as PDOSQLiteDriver;
use Doctrine\DBAL\Driver\PDOSqlsrv\Driver as PDOSQLSrvDriver;
use Doctrine\DBAL\Driver\SQLAnywhere\Driver as SQLAnywhereDriver;
use Doctrine\DBAL\Driver\SQLSrv\Driver as SQLSrvDriver;
17
use PDO;
18 19 20
use function array_keys;
use function array_map;
use function array_merge;
Sergei Morozov's avatar
Sergei Morozov committed
21
use function assert;
22 23
use function class_implements;
use function in_array;
Sergei Morozov's avatar
Sergei Morozov committed
24
use function is_string;
25 26 27 28 29 30 31
use function is_subclass_of;
use function parse_str;
use function parse_url;
use function preg_replace;
use function str_replace;
use function strpos;
use function substr;
romanb's avatar
romanb committed
32 33

/**
34
 * Factory for creating Doctrine\DBAL\Connection instances.
romanb's avatar
romanb committed
35
 */
36
final class DriverManager
romanb's avatar
romanb committed
37 38
{
    /**
39
     * List of supported drivers and their mappings to the driver classes.
romanb's avatar
romanb committed
40
     *
41 42 43
     * To add your own driver use the 'driverClass' parameter to
     * {@link DriverManager::getConnection()}.
     *
44
     * @var string[]
romanb's avatar
romanb committed
45
     */
46
    private static $_driverMap = [
47 48 49 50 51 52 53 54 55 56 57
        'pdo_mysql'          => PDOMySQLDriver::class,
        'pdo_sqlite'         => PDOSQLiteDriver::class,
        'pdo_pgsql'          => PDOPgSQLDriver::class,
        'pdo_oci'            => PDOOCIDriver::class,
        'oci8'               => OCI8Driver::class,
        'ibm_db2'            => DB2Driver::class,
        'pdo_sqlsrv'         => PDOSQLSrvDriver::class,
        'mysqli'             => MySQLiDriver::class,
        'drizzle_pdo_mysql'  => DrizzlePDOMySQLDriver::class,
        'sqlanywhere'        => SQLAnywhereDriver::class,
        'sqlsrv'             => SQLSrvDriver::class,
58
    ];
59

60 61
    /**
     * List of URL schemes from a database URL and their mappings to driver.
62 63
     *
     * @var string[]
64
     */
65
    private static $driverSchemeAliases = [
66 67 68 69 70 71 72 73 74
        'db2'        => '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',
75
    ];
76

Benjamin Morel's avatar
Benjamin Morel committed
77 78 79 80 81 82
    /**
     * Private constructor. This class cannot be instantiated.
     */
    private function __construct()
    {
    }
83

romanb's avatar
romanb committed
84
    /**
85
     * Creates a connection object based on the specified parameters.
86
     * This method returns a Doctrine\DBAL\Connection which wraps the underlying
87
     * driver connection.
romanb's avatar
romanb committed
88
     *
89
     * $params must contain at least one of the following.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
90
     *
91
     * Either 'driver' with one of the following values:
92
     *
93 94 95
     *     pdo_mysql
     *     pdo_sqlite
     *     pdo_pgsql
96 97
     *     pdo_oci (unstable)
     *     pdo_sqlsrv
98
     *     pdo_sqlsrv
99
     *     mysqli
100
     *     sqlanywhere
101 102 103
     *     sqlsrv
     *     ibm_db2 (unstable)
     *     drizzle_pdo_mysql
Benjamin Eberlei's avatar
Benjamin Eberlei committed
104
     *
105 106
     * OR 'driverClass' that contains the full class name (with namespace) of the
     * driver class to instantiate.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
107
     *
108
     * Other (optional) parameters:
Benjamin Eberlei's avatar
Benjamin Eberlei committed
109
     *
110
     * <b>user (string)</b>:
Benjamin Eberlei's avatar
Benjamin Eberlei committed
111 112
     * The username to use when connecting.
     *
113 114
     * <b>password (string)</b>:
     * The password to use when connecting.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
115
     *
116 117 118
     * <b>driverOptions (array)</b>:
     * Any additional driver-specific options for the driver. These are just passed
     * through to the driver.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
119
     *
120 121
     * <b>pdo</b>:
     * You can pass an existing PDO instance through this parameter. The PDO
122
     * instance will be wrapped in a Doctrine\DBAL\Connection.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
123
     *
124 125
     * <b>wrapperClass</b>:
     * You may specify a custom wrapper class through the 'wrapperClass'
126
     * parameter but this class MUST inherit from Doctrine\DBAL\Connection.
Benjamin Eberlei's avatar
Benjamin Eberlei committed
127
     *
128 129 130
     * <b>driverClass</b>:
     * The driver class to use.
     *
131
     * @param mixed[]            $params       The parameters.
132 133
     * @param Configuration|null $config       The configuration to use.
     * @param EventManager|null  $eventManager The event manager to use.
Benjamin Morel's avatar
Benjamin Morel committed
134
     *
135
     * @throws DBALException
romanb's avatar
romanb committed
136
     */
137
    public static function getConnection(
138 139 140 141
        array $params,
        ?Configuration $config = null,
        ?EventManager $eventManager = null
    ) : Connection {
romanb's avatar
romanb committed
142
        // create default config and event manager, if not set
143
        if (! $config) {
144
            $config = new Configuration();
romanb's avatar
romanb committed
145
        }
146
        if (! $eventManager) {
147
            $eventManager = new EventManager();
romanb's avatar
romanb committed
148
        }
Benjamin Eberlei's avatar
Benjamin Eberlei committed
149

David Zuelke's avatar
David Zuelke committed
150
        $params = self::parseDatabaseUrl($params);
151

152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173
        // 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);
            }
        }

romanb's avatar
romanb committed
174
        // check for existing pdo object
175
        if (isset($params['pdo']) && ! $params['pdo'] instanceof PDO) {
176
            throw DBALException::invalidPdoInstance();
177 178 179
        }

        if (isset($params['pdo'])) {
180 181
            $params['pdo']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
            $params['driver'] = 'pdo_' . $params['pdo']->getAttribute(PDO::ATTR_DRIVER_NAME);
romanb's avatar
romanb committed
182
        } else {
183
            self::_checkParams($params);
romanb's avatar
romanb committed
184
        }
185

186
        $className = $params['driverClass'] ?? self::$_driverMap[$params['driver']];
Benjamin Eberlei's avatar
Benjamin Eberlei committed
187

romanb's avatar
romanb committed
188
        $driver = new $className();
Benjamin Eberlei's avatar
Benjamin Eberlei committed
189

190
        $wrapperClass = Connection::class;
191
        if (isset($params['wrapperClass'])) {
192
            if (! is_subclass_of($params['wrapperClass'], $wrapperClass)) {
193 194
                throw DBALException::invalidWrapperClass($params['wrapperClass']);
            }
195 196

            $wrapperClass = $params['wrapperClass'];
romanb's avatar
romanb committed
197
        }
Benjamin Eberlei's avatar
Benjamin Eberlei committed
198

romanb's avatar
romanb committed
199 200
        return new $wrapperClass($params, $driver, $config, $eventManager);
    }
201

202
    /**
203
     * Returns the list of supported drivers.
204
     *
205
     * @return string[]
206
     */
207
    public static function getAvailableDrivers() : array
208
    {
209
        return array_keys(self::$_driverMap);
210 211
    }

romanb's avatar
romanb committed
212 213 214
    /**
     * Checks the list of parameters.
     *
215
     * @param mixed[] $params The list of parameters.
Benjamin Morel's avatar
Benjamin Morel committed
216
     *
217
     * @throws DBALException
romanb's avatar
romanb committed
218
     */
219
    private static function _checkParams(array $params) : void
Benjamin Eberlei's avatar
Benjamin Eberlei committed
220
    {
Pascal Borreli's avatar
Pascal Borreli committed
221
        // check existence of mandatory parameters
Benjamin Eberlei's avatar
Benjamin Eberlei committed
222

romanb's avatar
romanb committed
223
        // driver
224
        if (! isset($params['driver']) && ! isset($params['driverClass'])) {
225
            throw DBALException::driverRequired();
romanb's avatar
romanb committed
226
        }
Benjamin Eberlei's avatar
Benjamin Eberlei committed
227

romanb's avatar
romanb committed
228
        // check validity of parameters
Benjamin Eberlei's avatar
Benjamin Eberlei committed
229

romanb's avatar
romanb committed
230
        // driver
231
        if (isset($params['driver']) && ! isset(self::$_driverMap[$params['driver']])) {
232
            throw DBALException::unknownDriver($params['driver'], array_keys(self::$_driverMap));
romanb's avatar
romanb committed
233
        }
234

235
        if (isset($params['driverClass']) && ! in_array(Driver::class, class_implements($params['driverClass'], true))) {
236 237
            throw DBALException::invalidDriverClass($params['driverClass']);
        }
romanb's avatar
romanb committed
238
    }
239

240 241 242 243 244
    /**
     * Normalizes the given connection URL path.
     *
     * @return string The normalized connection URL path
     */
245
    private static function normalizeDatabaseUrlPath(string $urlPath) : string
246 247 248 249 250
    {
        // Trim leading slash from URL path.
        return substr($urlPath, 1);
    }

251 252 253 254
    /**
     * Extracts parts from a database URL, if present, and returns an
     * updated list of parameters.
     *
255
     * @param mixed[] $params The list of parameters.
256
     *
257 258
     * @return mixed[] A modified list of parameters with info from a database
     *                 URL extracted into indidivual parameter parts.
259
     *
260
     * @throws DBALException
261
     */
262
    private static function parseDatabaseUrl(array $params) : array
263
    {
264
        if (! isset($params['url'])) {
David Zuelke's avatar
David Zuelke committed
265
            return $params;
266
        }
267

268
        // (pdo_)?sqlite3?:///... => (pdo_)?sqlite3?://localhost/... or else the URL will be invalid
David Zuelke's avatar
David Zuelke committed
269
        $url = preg_replace('#^((?:pdo_)?sqlite3?):///#', '$1://localhost/', $params['url']);
Sergei Morozov's avatar
Sergei Morozov committed
270 271
        assert(is_string($url));

272
        $url = parse_url($url);
273

David Zuelke's avatar
David Zuelke committed
274
        if ($url === false) {
275 276
            throw new DBALException('Malformed parameter "url".');
        }
277

278 279
        $url = array_map('rawurldecode', $url);

280 281 282 283 284
        // If we have a connection URL, we have to unset the default PDO instance connection parameter (if any)
        // as we cannot merge connection details from the URL into the PDO instance (URL takes precedence).
        unset($params['pdo']);

        $params = self::parseDatabaseUrlScheme($url, $params);
285

286 287 288 289 290 291 292 293 294 295 296 297
        if (isset($url['host'])) {
            $params['host'] = $url['host'];
        }
        if (isset($url['port'])) {
            $params['port'] = $url['port'];
        }
        if (isset($url['user'])) {
            $params['user'] = $url['user'];
        }
        if (isset($url['pass'])) {
            $params['password'] = $url['pass'];
        }
298

299 300
        $params = self::parseDatabaseUrlPath($url, $params);
        $params = self::parseDatabaseUrlQuery($url, $params);
301 302 303 304 305

        return $params;
    }

    /**
306 307 308 309
     * Parses the given connection URL and resolves the given connection parameters.
     *
     * Assumes that the connection URL scheme is already parsed and resolved into the given connection parameters
     * via {@link parseDatabaseUrlScheme}.
310
     *
311 312
     * @see parseDatabaseUrlScheme
     *
313 314
     * @param mixed[] $url    The URL parts to evaluate.
     * @param mixed[] $params The connection parameters to resolve.
315
     *
316
     * @return mixed[] The resolved connection parameters.
317
     */
318
    private static function parseDatabaseUrlPath(array $url, array $params) : array
319
    {
320
        if (! isset($url['path'])) {
321 322 323
            return $params;
        }

324
        $url['path'] = self::normalizeDatabaseUrlPath($url['path']);
325

326 327 328 329 330 331 332
        // If we do not have a known DBAL driver, we do not know any connection URL path semantics to evaluate
        // and therefore treat the path as regular DBAL connection URL path.
        if (! isset($params['driver'])) {
            return self::parseRegularDatabaseUrlPath($url, $params);
        }

        if (strpos($params['driver'], 'sqlite') !== false) {
333 334 335
            return self::parseSqliteDatabaseUrlPath($url, $params);
        }

336 337 338 339 340 341
        return self::parseRegularDatabaseUrlPath($url, $params);
    }

    /**
     * Parses the query part of the given connection URL and resolves the given connection parameters.
     *
342 343
     * @param mixed[] $url    The connection URL parts to evaluate.
     * @param mixed[] $params The connection parameters to resolve.
344
     *
345
     * @return mixed[] The resolved connection parameters.
346
     */
347
    private static function parseDatabaseUrlQuery(array $url, array $params) : array
348 349 350 351 352
    {
        if (! isset($url['query'])) {
            return $params;
        }

353
        $query = [];
354 355 356 357 358 359 360 361 362 363 364

        parse_str($url['query'], $query); // simply ingest query as extra params, e.g. charset or sslmode

        return array_merge($params, $query); // parse_str wipes existing array elements
    }

    /**
     * Parses the given regular connection URL and resolves the given connection parameters.
     *
     * Assumes that the "path" URL part is already normalized via {@link normalizeDatabaseUrlPath}.
     *
365 366
     * @see normalizeDatabaseUrlPath
     *
367 368
     * @param mixed[] $url    The regular connection URL parts to evaluate.
     * @param mixed[] $params The connection parameters to resolve.
369
     *
370
     * @return mixed[] The resolved connection parameters.
371
     */
372
    private static function parseRegularDatabaseUrlPath(array $url, array $params) : array
373
    {
374 375 376 377 378 379
        $params['dbname'] = $url['path'];

        return $params;
    }

    /**
380
     * Parses the given SQLite connection URL and resolves the given connection parameters.
381
     *
382 383
     * Assumes that the "path" URL part is already normalized via {@link normalizeDatabaseUrlPath}.
     *
384 385
     * @see normalizeDatabaseUrlPath
     *
386 387
     * @param mixed[] $url    The SQLite connection URL parts to evaluate.
     * @param mixed[] $params The connection parameters to resolve.
388
     *
389
     * @return mixed[] The resolved connection parameters.
390
     */
391
    private static function parseSqliteDatabaseUrlPath(array $url, array $params) : array
392 393 394 395 396 397 398 399 400
    {
        if ($url['path'] === ':memory:') {
            $params['memory'] = true;

            return $params;
        }

        $params['path'] = $url['path']; // pdo_sqlite driver uses 'path' instead of 'dbname' key

David Zuelke's avatar
David Zuelke committed
401
        return $params;
402
    }
403 404 405 406

    /**
     * Parses the scheme part from given connection URL and resolves the given connection parameters.
     *
407 408
     * @param mixed[] $url    The connection URL parts to evaluate.
     * @param mixed[] $params The connection parameters to resolve.
409
     *
410
     * @return mixed[] The resolved connection parameters.
411
     *
412
     * @throws DBALException If parsing failed or resolution is not possible.
413
     */
414
    private static function parseDatabaseUrlScheme(array $url, array $params) : array
415 416 417 418 419 420 421 422
    {
        if (isset($url['scheme'])) {
            // The requested driver from the URL scheme takes precedence
            // over the default custom driver from the connection parameters (if any).
            unset($params['driverClass']);

            // URL schemes must not contain underscores, but dashes are ok
            $driver = str_replace('-', '_', $url['scheme']);
Sergei Morozov's avatar
Sergei Morozov committed
423
            assert(is_string($driver));
424

425 426 427 428
            // The requested driver from the URL scheme takes precedence over the
            // default driver from the connection parameters. If the driver is
            // an alias (e.g. "postgres"), map it to the actual name ("pdo-pgsql").
            // Otherwise, let checkParams decide later if the driver exists.
429
            $params['driver'] = self::$driverSchemeAliases[$driver] ?? $driver;
430 431 432 433 434 435 436 437 438 439 440 441

            return $params;
        }

        // 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 DBALException::driverRequired($params['url']);
        }

        return $params;
    }
Benjamin Eberlei's avatar
Benjamin Eberlei committed
442
}