From 59aca8ed4793bf79a934701008189c843e486ec6 Mon Sep 17 00:00:00 2001 From: kenjis Date: Fri, 30 Aug 2024 09:53:58 +0900 Subject: [PATCH 1/2] feat: add function to check PHP DB extension To improve the error message. --- system/Database/Database.php | 44 +++++++++++++++++++++++++++++++++--- 1 file changed, 41 insertions(+), 3 deletions(-) diff --git a/system/Database/Database.php b/system/Database/Database.php index c2bc66d30d11..5fabda87d77c 100644 --- a/system/Database/Database.php +++ b/system/Database/Database.php @@ -13,6 +13,8 @@ namespace CodeIgniter\Database; +use CodeIgniter\Exceptions\ConfigException; +use CodeIgniter\Exceptions\CriticalError; use InvalidArgumentException; /** @@ -54,6 +56,8 @@ public function load(array $params = [], string $alias = '') throw new InvalidArgumentException('You have not selected a database type to connect to.'); } + assert($this->checkDbExtension($params['DBDriver'])); + $this->connections[$alias] = $this->initDriver($params['DBDriver'], 'Connection', $params); return $this->connections[$alias]; @@ -124,9 +128,9 @@ protected function parseDSN(array $params): array /** * Creates a database object. * - * @param string $driver Driver name. FQCN can be used. - * @param string $class 'Connection'|'Forge'|'Utils' - * @param array|object $argument The constructor parameter. + * @param string $driver Driver name. FQCN can be used. + * @param string $class 'Connection'|'Forge'|'Utils' + * @param array|ConnectionInterface $argument The constructor parameter or DB connection * * @return BaseConnection|BaseUtils|Forge */ @@ -138,4 +142,38 @@ protected function initDriver(string $driver, string $class, $argument): object return new $classname($argument); } + + /** + * Check the PHP database extension is loaded. + * + * @param string $driver DB driver + */ + private function checkDbExtension(string $driver): bool + { + $extensionMap = [ + // DBDriver => PHP extension + 'MySQLi' => 'mysqli', + 'SQLite3' => 'sqlite3', + 'Postgre' => 'pgsql', + 'SQLSRV' => 'sqlsrv', + 'OCI8' => 'oci8', + ]; + + $extension = $extensionMap[$driver] ?? ''; + + if ($extension === '') { + $message = 'Invalid DBDriver name: "' . $driver . '"'; + + throw new ConfigException($message); + } + + if (extension_loaded($extension)) { + return true; + } + + $message = 'The required PHP extension "' . $extension . '" is not loaded.' + . ' Install and enable it to use "' . $driver . '" driver.'; + + throw new CriticalError($message); + } } From a9925c740e180c96c25bac5fbb4ec90005eacc76 Mon Sep 17 00:00:00 2001 From: kenjis Date: Fri, 30 Aug 2024 10:22:55 +0900 Subject: [PATCH 2/2] fix: pass through custom drivers Because we can name any classname. --- system/Database/Database.php | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/system/Database/Database.php b/system/Database/Database.php index 5fabda87d77c..23b6cd0f8d16 100644 --- a/system/Database/Database.php +++ b/system/Database/Database.php @@ -146,10 +146,15 @@ protected function initDriver(string $driver, string $class, $argument): object /** * Check the PHP database extension is loaded. * - * @param string $driver DB driver + * @param string $driver DB driver or FQCN for custom driver */ private function checkDbExtension(string $driver): bool { + if (str_contains($driver, '\\')) { + // Cannot check a fully qualified classname for a custom driver. + return true; + } + $extensionMap = [ // DBDriver => PHP extension 'MySQLi' => 'mysqli',