mysqli_driver.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. <?php
  2. /**
  3. * CodeIgniter
  4. *
  5. * An open source application development framework for PHP
  6. *
  7. * This content is released under the MIT License (MIT)
  8. *
  9. * Copyright (c) 2014 - 2019, British Columbia Institute of Technology
  10. *
  11. * Permission is hereby granted, free of charge, to any person obtaining a copy
  12. * of this software and associated documentation files (the "Software"), to deal
  13. * in the Software without restriction, including without limitation the rights
  14. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  15. * copies of the Software, and to permit persons to whom the Software is
  16. * furnished to do so, subject to the following conditions:
  17. *
  18. * The above copyright notice and this permission notice shall be included in
  19. * all copies or substantial portions of the Software.
  20. *
  21. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  22. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  23. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  24. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  25. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  26. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  27. * THE SOFTWARE.
  28. *
  29. * @package CodeIgniter
  30. * @author EllisLab Dev Team
  31. * @copyright Copyright (c) 2008 - 2014, EllisLab, Inc. (https://ellislab.com/)
  32. * @copyright Copyright (c) 2014 - 2019, British Columbia Institute of Technology (https://bcit.ca/)
  33. * @license https://opensource.org/licenses/MIT MIT License
  34. * @link https://codeigniter.com
  35. * @since Version 1.3.0
  36. * @filesource
  37. */
  38. defined('BASEPATH') OR exit('No direct script access allowed');
  39. /**
  40. * MySQLi Database Adapter Class
  41. *
  42. * Note: _DB is an extender class that the app controller
  43. * creates dynamically based on whether the query builder
  44. * class is being used or not.
  45. *
  46. * @package CodeIgniter
  47. * @subpackage Drivers
  48. * @category Database
  49. * @author EllisLab Dev Team
  50. * @link https://codeigniter.com/user_guide/database/
  51. */
  52. class CI_DB_mysqli_driver extends CI_DB {
  53. /**
  54. * Database driver
  55. *
  56. * @var string
  57. */
  58. public $dbdriver = 'mysqli';
  59. /**
  60. * Compression flag
  61. *
  62. * @var bool
  63. */
  64. public $compress = FALSE;
  65. /**
  66. * DELETE hack flag
  67. *
  68. * Whether to use the MySQL "delete hack" which allows the number
  69. * of affected rows to be shown. Uses a preg_replace when enabled,
  70. * adding a bit more processing to all queries.
  71. *
  72. * @var bool
  73. */
  74. public $delete_hack = TRUE;
  75. /**
  76. * Strict ON flag
  77. *
  78. * Whether we're running in strict SQL mode.
  79. *
  80. * @var bool
  81. */
  82. public $stricton;
  83. // --------------------------------------------------------------------
  84. /**
  85. * Identifier escape character
  86. *
  87. * @var string
  88. */
  89. protected $_escape_char = '`';
  90. // --------------------------------------------------------------------
  91. /**
  92. * MySQLi object
  93. *
  94. * Has to be preserved without being assigned to $conn_id.
  95. *
  96. * @var MySQLi
  97. */
  98. protected $_mysqli;
  99. // --------------------------------------------------------------------
  100. /**
  101. * Database connection
  102. *
  103. * @param bool $persistent
  104. * @return object
  105. */
  106. public function db_connect($persistent = FALSE)
  107. {
  108. // Do we have a socket path?
  109. if ($this->hostname[0] === '/')
  110. {
  111. $hostname = NULL;
  112. $port = NULL;
  113. $socket = $this->hostname;
  114. }
  115. else
  116. {
  117. $hostname = ($persistent === TRUE)
  118. ? 'p:'.$this->hostname : $this->hostname;
  119. $port = empty($this->port) ? NULL : $this->port;
  120. $socket = NULL;
  121. }
  122. $client_flags = ($this->compress === TRUE) ? MYSQLI_CLIENT_COMPRESS : 0;
  123. $this->_mysqli = mysqli_init();
  124. $this->_mysqli->options(MYSQLI_OPT_CONNECT_TIMEOUT, 10);
  125. if (isset($this->stricton))
  126. {
  127. if ($this->stricton)
  128. {
  129. $this->_mysqli->options(MYSQLI_INIT_COMMAND, 'SET SESSION sql_mode = CONCAT(@@sql_mode, ",", "STRICT_ALL_TABLES")');
  130. }
  131. else
  132. {
  133. $this->_mysqli->options(MYSQLI_INIT_COMMAND,
  134. 'SET SESSION sql_mode =
  135. REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(
  136. @@sql_mode,
  137. "STRICT_ALL_TABLES,", ""),
  138. ",STRICT_ALL_TABLES", ""),
  139. "STRICT_ALL_TABLES", ""),
  140. "STRICT_TRANS_TABLES,", ""),
  141. ",STRICT_TRANS_TABLES", ""),
  142. "STRICT_TRANS_TABLES", "")'
  143. );
  144. }
  145. }
  146. if (is_array($this->encrypt))
  147. {
  148. $ssl = array();
  149. empty($this->encrypt['ssl_key']) OR $ssl['key'] = $this->encrypt['ssl_key'];
  150. empty($this->encrypt['ssl_cert']) OR $ssl['cert'] = $this->encrypt['ssl_cert'];
  151. empty($this->encrypt['ssl_ca']) OR $ssl['ca'] = $this->encrypt['ssl_ca'];
  152. empty($this->encrypt['ssl_capath']) OR $ssl['capath'] = $this->encrypt['ssl_capath'];
  153. empty($this->encrypt['ssl_cipher']) OR $ssl['cipher'] = $this->encrypt['ssl_cipher'];
  154. if (isset($this->encrypt['ssl_verify']))
  155. {
  156. $client_flags |= MYSQLI_CLIENT_SSL;
  157. if ($this->encrypt['ssl_verify'])
  158. {
  159. defined('MYSQLI_OPT_SSL_VERIFY_SERVER_CERT') && $this->_mysqli->options(MYSQLI_OPT_SSL_VERIFY_SERVER_CERT, TRUE);
  160. }
  161. // Apparently (when it exists), setting MYSQLI_OPT_SSL_VERIFY_SERVER_CERT
  162. // to FALSE didn't do anything, so PHP 5.6.16 introduced yet another
  163. // constant ...
  164. //
  165. // https://secure.php.net/ChangeLog-5.php#5.6.16
  166. // https://bugs.php.net/bug.php?id=68344
  167. elseif (defined('MYSQLI_CLIENT_SSL_DONT_VERIFY_SERVER_CERT'))
  168. {
  169. $client_flags |= MYSQLI_CLIENT_SSL_DONT_VERIFY_SERVER_CERT;
  170. }
  171. }
  172. if ( ! empty($ssl))
  173. {
  174. $client_flags |= MYSQLI_CLIENT_SSL;
  175. $this->_mysqli->ssl_set(
  176. isset($ssl['key']) ? $ssl['key'] : NULL,
  177. isset($ssl['cert']) ? $ssl['cert'] : NULL,
  178. isset($ssl['ca']) ? $ssl['ca'] : NULL,
  179. isset($ssl['capath']) ? $ssl['capath'] : NULL,
  180. isset($ssl['cipher']) ? $ssl['cipher'] : NULL
  181. );
  182. }
  183. }
  184. if ($this->_mysqli->real_connect($hostname, $this->username, $this->password, $this->database, $port, $socket, $client_flags))
  185. {
  186. // Prior to version 5.7.3, MySQL silently downgrades to an unencrypted connection if SSL setup fails
  187. if (
  188. ($client_flags & MYSQLI_CLIENT_SSL)
  189. && version_compare($this->_mysqli->client_info, '5.7.3', '<=')
  190. && empty($this->_mysqli->query("SHOW STATUS LIKE 'ssl_cipher'")->fetch_object()->Value)
  191. )
  192. {
  193. $this->_mysqli->close();
  194. $message = 'MySQLi was configured for an SSL connection, but got an unencrypted connection instead!';
  195. log_message('error', $message);
  196. return ($this->db_debug) ? $this->display_error($message, '', TRUE) : FALSE;
  197. }
  198. return $this->_mysqli;
  199. }
  200. return FALSE;
  201. }
  202. // --------------------------------------------------------------------
  203. /**
  204. * Reconnect
  205. *
  206. * Keep / reestablish the db connection if no queries have been
  207. * sent for a length of time exceeding the server's idle timeout
  208. *
  209. * @return void
  210. */
  211. public function reconnect()
  212. {
  213. if ($this->conn_id !== FALSE && $this->conn_id->ping() === FALSE)
  214. {
  215. $this->conn_id = FALSE;
  216. }
  217. }
  218. // --------------------------------------------------------------------
  219. /**
  220. * Select the database
  221. *
  222. * @param string $database
  223. * @return bool
  224. */
  225. public function db_select($database = '')
  226. {
  227. if ($database === '')
  228. {
  229. $database = $this->database;
  230. }
  231. if ($this->conn_id->select_db($database))
  232. {
  233. $this->database = $database;
  234. $this->data_cache = array();
  235. return TRUE;
  236. }
  237. return FALSE;
  238. }
  239. // --------------------------------------------------------------------
  240. /**
  241. * Set client character set
  242. *
  243. * @param string $charset
  244. * @return bool
  245. */
  246. protected function _db_set_charset($charset)
  247. {
  248. return $this->conn_id->set_charset($charset);
  249. }
  250. // --------------------------------------------------------------------
  251. /**
  252. * Database version number
  253. *
  254. * @return string
  255. */
  256. public function version()
  257. {
  258. if (isset($this->data_cache['version']))
  259. {
  260. return $this->data_cache['version'];
  261. }
  262. return $this->data_cache['version'] = $this->conn_id->server_info;
  263. }
  264. // --------------------------------------------------------------------
  265. /**
  266. * Execute the query
  267. *
  268. * @param string $sql an SQL query
  269. * @return mixed
  270. */
  271. protected function _execute($sql)
  272. {
  273. return $this->conn_id->query($this->_prep_query($sql));
  274. }
  275. // --------------------------------------------------------------------
  276. /**
  277. * Prep the query
  278. *
  279. * If needed, each database adapter can prep the query string
  280. *
  281. * @param string $sql an SQL query
  282. * @return string
  283. */
  284. protected function _prep_query($sql)
  285. {
  286. // mysqli_affected_rows() returns 0 for "DELETE FROM TABLE" queries. This hack
  287. // modifies the query so that it a proper number of affected rows is returned.
  288. if ($this->delete_hack === TRUE && preg_match('/^\s*DELETE\s+FROM\s+(\S+)\s*$/i', $sql))
  289. {
  290. return trim($sql).' WHERE 1=1';
  291. }
  292. return $sql;
  293. }
  294. // --------------------------------------------------------------------
  295. /**
  296. * Begin Transaction
  297. *
  298. * @return bool
  299. */
  300. protected function _trans_begin()
  301. {
  302. $this->conn_id->autocommit(FALSE);
  303. return is_php('5.5')
  304. ? $this->conn_id->begin_transaction()
  305. : $this->simple_query('START TRANSACTION'); // can also be BEGIN or BEGIN WORK
  306. }
  307. // --------------------------------------------------------------------
  308. /**
  309. * Commit Transaction
  310. *
  311. * @return bool
  312. */
  313. protected function _trans_commit()
  314. {
  315. if ($this->conn_id->commit())
  316. {
  317. $this->conn_id->autocommit(TRUE);
  318. return TRUE;
  319. }
  320. return FALSE;
  321. }
  322. // --------------------------------------------------------------------
  323. /**
  324. * Rollback Transaction
  325. *
  326. * @return bool
  327. */
  328. protected function _trans_rollback()
  329. {
  330. if ($this->conn_id->rollback())
  331. {
  332. $this->conn_id->autocommit(TRUE);
  333. return TRUE;
  334. }
  335. return FALSE;
  336. }
  337. // --------------------------------------------------------------------
  338. /**
  339. * Platform-dependent string escape
  340. *
  341. * @param string
  342. * @return string
  343. */
  344. protected function _escape_str($str)
  345. {
  346. return $this->conn_id->real_escape_string($str);
  347. }
  348. // --------------------------------------------------------------------
  349. /**
  350. * Affected Rows
  351. *
  352. * @return int
  353. */
  354. public function affected_rows()
  355. {
  356. return $this->conn_id->affected_rows;
  357. }
  358. // --------------------------------------------------------------------
  359. /**
  360. * Insert ID
  361. *
  362. * @return int
  363. */
  364. public function insert_id()
  365. {
  366. return $this->conn_id->insert_id;
  367. }
  368. // --------------------------------------------------------------------
  369. /**
  370. * List table query
  371. *
  372. * Generates a platform-specific query string so that the table names can be fetched
  373. *
  374. * @param bool $prefix_limit
  375. * @return string
  376. */
  377. protected function _list_tables($prefix_limit = FALSE)
  378. {
  379. $sql = 'SHOW TABLES FROM '.$this->_escape_char.$this->database.$this->_escape_char;
  380. if ($prefix_limit !== FALSE && $this->dbprefix !== '')
  381. {
  382. return $sql." LIKE '".$this->escape_like_str($this->dbprefix)."%'";
  383. }
  384. return $sql;
  385. }
  386. // --------------------------------------------------------------------
  387. /**
  388. * Show column query
  389. *
  390. * Generates a platform-specific query string so that the column names can be fetched
  391. *
  392. * @param string $table
  393. * @return string
  394. */
  395. protected function _list_columns($table = '')
  396. {
  397. return 'SHOW COLUMNS FROM '.$this->protect_identifiers($table, TRUE, NULL, FALSE);
  398. }
  399. // --------------------------------------------------------------------
  400. /**
  401. * Returns an object with field data
  402. *
  403. * @param string $table
  404. * @return array
  405. */
  406. public function field_data($table)
  407. {
  408. if (($query = $this->query('SHOW COLUMNS FROM '.$this->protect_identifiers($table, TRUE, NULL, FALSE))) === FALSE)
  409. {
  410. return FALSE;
  411. }
  412. $query = $query->result_object();
  413. $retval = array();
  414. for ($i = 0, $c = count($query); $i < $c; $i++)
  415. {
  416. $retval[$i] = new stdClass();
  417. $retval[$i]->name = $query[$i]->Field;
  418. sscanf($query[$i]->Type, '%[a-z](%d)',
  419. $retval[$i]->type,
  420. $retval[$i]->max_length
  421. );
  422. $retval[$i]->default = $query[$i]->Default;
  423. $retval[$i]->primary_key = (int) ($query[$i]->Key === 'PRI');
  424. }
  425. return $retval;
  426. }
  427. // --------------------------------------------------------------------
  428. /**
  429. * Error
  430. *
  431. * Returns an array containing code and message of the last
  432. * database error that has occurred.
  433. *
  434. * @return array
  435. */
  436. public function error()
  437. {
  438. if ( ! empty($this->_mysqli->connect_errno))
  439. {
  440. return array(
  441. 'code' => $this->_mysqli->connect_errno,
  442. 'message' => $this->_mysqli->connect_error
  443. );
  444. }
  445. return array('code' => $this->conn_id->errno, 'message' => $this->conn_id->error);
  446. }
  447. // --------------------------------------------------------------------
  448. /**
  449. * FROM tables
  450. *
  451. * Groups tables in FROM clauses if needed, so there is no confusion
  452. * about operator precedence.
  453. *
  454. * @return string
  455. */
  456. protected function _from_tables()
  457. {
  458. if ( ! empty($this->qb_join) && count($this->qb_from) > 1)
  459. {
  460. return '('.implode(', ', $this->qb_from).')';
  461. }
  462. return implode(', ', $this->qb_from);
  463. }
  464. // --------------------------------------------------------------------
  465. /**
  466. * Close DB Connection
  467. *
  468. * @return void
  469. */
  470. protected function _close()
  471. {
  472. $this->conn_id->close();
  473. }
  474. }