Create New Item
Item Type
File
Folder
Item Name
Search file in folder and subfolders...
Are you sure want to rename?
File Manager
/
ancien-site-2019
/
site
/
libraries
/
joomla
/
database
/
database
:
mysqli.php
Advanced Search
Upload
New Item
Settings
Back
Back Up
Advanced Editor
Save
<?php /** * @package Joomla.Platform * @subpackage Database * * @copyright Copyright (C) 2005 - 2011 Open Source Matters, Inc. All rights reserved. * @license GNU General Public License version 2 or later; see LICENSE */ defined('JPATH_PLATFORM') or die; JLoader::register('JDatabaseQueryMySQLi', dirname(__FILE__).'/mysqliquery.php'); JLoader::register('JDatabaseExporterMySQLi', dirname(__FILE__).'/mysqliexporter.php'); JLoader::register('JDatabaseImporterMySQLi', dirname(__FILE__).'/mysqliimporter.php'); /** * MySQLi database driver * * @package Joomla.Platform * @subpackage Database * @see http://php.net/manual/en/book.mysqli.php * @since 11.1 */ class JDatabaseMySQLi extends JDatabase { /** * The name of the database driver. * * @var string * @since 11.1 */ public $name = 'mysqli'; /** * The character(s) used to quote SQL statement names such as table names or field names, * etc. The child classes should define this as necessary. If a single character string the * same character is used for both sides of the quoted name, else the first character will be * used for the opening quote and the second for the closing quote. * * @var string * @since 11.1 */ protected $nameQuote = '`'; /** * The null or zero representation of a timestamp for the database driver. This should be * defined in child classes to hold the appropriate value for the engine. * * @var string * @since 11.1 */ protected $nullDate = '0000-00-00 00:00:00'; /** * Constructor. * * @param array $options List of options used to configure the connection * * @return void * * @since 11.1 */ protected function __construct($options) { // Get some basic values from the options. $options['host'] = (isset($options['host'])) ? $options['host'] : 'localhost'; $options['user'] = (isset($options['user'])) ? $options['user'] : 'root'; $options['password'] = (isset($options['password'])) ? $options['password'] : ''; $options['database'] = (isset($options['database'])) ? $options['database'] : ''; $options['select'] = (isset($options['select'])) ? (bool) $options['select'] : true; $options['port'] = null; $options['socket'] = null; /* * Unlike mysql_connect(), mysqli_connect() takes the port and socket as separate arguments. Therefore, we * have to extract them from the host string. */ $tmp = substr(strstr($options['host'], ':'), 1); if (!empty($tmp)) { // Get the port number or socket name if (is_numeric($tmp)) { $options['port'] = $tmp; } else { $options['socket'] = $tmp; } // Extract the host name only $options['host'] = substr($options['host'], 0, strlen($options['host']) - (strlen($tmp) + 1)); // This will take care of the following notation: ":3306" if ($options['host'] == '') { $options['host'] = 'localhost'; } } // Make sure the MySQLi extension for PHP is installed and enabled. if (!function_exists('mysqli_connect')) { // Legacy error handling switch based on the JError::$legacy switch. // @deprecated 12.1 if (JError::$legacy) { $this->errorNum = 1; $this->errorMsg = JText::_('JLIB_DATABASE_ERROR_ADAPTER_MYSQLI'); return; } else { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_ADAPTER_MYSQLI')); } } // Attempt to connect to the server. if (!($this->connection = @ mysqli_connect($options['host'], $options['user'], $options['password'], null, $options['port'], $options['socket']))) { // Legacy error handling switch based on the JError::$legacy switch. // @deprecated 12.1 if (JError::$legacy) { $this->errorNum = 2; $this->errorMsg = JText::_('JLIB_DATABASE_ERROR_CONNECT_MYSQL'); return; } else { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_CONNECT_MYSQL')); } } // Finalize initialisation parent::__construct($options); // Set sql_mode to non_strict mode mysqli_query($this->connection, "SET @@SESSION.sql_mode = '';"); // If auto-select is enabled select the given database. if ($options['select'] && !empty($options['database'])) { $this->select($options['database']); } } /** * Destructor. * * @return void * * @since 11.1 */ public function __destruct() { if (is_object($this->connection)) { mysqli_close($this->connection); } } /** * Method to escape a string for usage in an SQL statement. * * @param string $text The string to be escaped. * @param bool $extra Optional parameter to provide extra escaping. * * @return string The escaped string. * * @since 11.1 */ public function escape($text, $extra = false) { $result = mysqli_real_escape_string($this->getConnection(), $text); if ($extra) { $result = addcslashes($result, '%_'); } return $result; } /** * Test to see if the MySQL connector is available. * * @return boolean True on success, false otherwise. * * @since 11.1 */ public static function test() { return (function_exists('mysqli_connect')); } /** * Determines if the connection to the server is active. * * @return boolean True if connected to the database engine. * * @since 11.1 */ public function connected() { return $this->connection->ping(); } /** * Drops a table from the database. * * @param string $tableName The name of the database table to drop. * @param boolean $ifExists Optionally specify that the table must exist before it is dropped. * * @return JDatabaseSQLSrv Returns this object to support chaining. * @since 11.1 */ function dropTable($tableName, $ifExists = true) { $query = $this->getQuery(true); $this->setQuery( 'DROP TABLE '. ($ifExists ? 'IF EXISTS ' : ''). $query->quoteName($tableName) ); $this->query(); return $this; } /** * Get the number of affected rows for the previous executed SQL statement. * * @return integer The number of affected rows. * * @since 11.1 */ public function getAffectedRows() { return mysqli_affected_rows($this->connection); } /** * Method to get the database collation in use by sampling a text field of a table in the database. * * @return mixed The collation in use by the database or boolean false if not supported. * * @since 11.1 */ public function getCollation() { if ($this->hasUTF()) { $this->setQuery('SHOW FULL COLUMNS FROM #__users'); $array = $this->loadAssocList(); return $array['2']['Collation']; } else { return 'N/A (Not Able to Detect)'; } } /** * Gets an exporter class object. * * @return JDatabaseExporterMySQLi An exporter object. * * @since 11.1 * @throws DatabaseException */ public function getExporter() { // Make sure we have an exporter class for this driver. if (!class_exists('JDatabaseExporterMySQLi')) { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_MISSING_EXPORTER')); } $o = new JDatabaseExporterMySQLi; $o->setDbo($this); return $o; } /** * Gets an importer class object. * * @return JDatabaseImporterMySQLi An importer object. * * @since 11.1 * @throws DatabaseException */ public function getImporter() { // Make sure we have an importer class for this driver. if (!class_exists('JDatabaseImporterMySQLi')) { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_MISSING_IMPORTER')); } $o = new JDatabaseImporterMySQLi; $o->setDbo($this); return $o; } /** * Get the number of returned rows for the previous executed SQL statement. * * @param resource $cursor An optional database cursor resource to extract the row count from. * * @return integer The number of returned rows. * * @since 11.1 */ public function getNumRows($cursor = null) { return mysqli_num_rows($cursor ? $cursor : $this->cursor); } /** * Get the current or query, or new JDatabaseQuery object. * * @param boolean $new False to return the last query set, True to return a new JDatabaseQuery object. * * @return mixed The current value of the internal SQL variable or a new JDatabaseQuery object. * * @since 11.1 * @throws DatabaseException */ function getQuery($new = false) { if ($new) { // Make sure we have a query class for this driver. if (!class_exists('JDatabaseQueryMySQLi')) { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_MISSING_QUERY')); } return new JDatabaseQueryMySQLi($this); } else { return $this->sql; } } /** * Shows the table CREATE statement that creates the given tables. * * @param mixed $tables A table name or a list of table names. * * @return array A list of the create SQL for the tables. * * @since 11.1 * @throws DatabaseException */ public function getTableCreate($tables) { // Initialise variables. $result = array(); // Sanitize input to an array and iterate over the list. settype($tables, 'array'); foreach ($tables as $table) { // Set the query to get the table CREATE statement. $this->setQuery('SHOW CREATE table '.$this->quoteName($this->escape($table))); $row = $this->loadRow(); // Populate the result array based on the create statements. $result[$table] = $row[1]; } return $result; } /** * Retrieves field information about a given table. * * @param string $table The name of the database table. * @param bool $typeOnly True to only return field types. * * @return array An array of fields for the database table. * * @since 11.1 * @throws DatabaseException */ public function getTableColumns($table, $typeOnly = true) { $result = array(); $query = $this->getQuery(true); // Set the query to get the table fields statement. $this->setQuery('SHOW FULL COLUMNS FROM '.$this->quoteName($this->escape($table))); $fields = $this->loadObjectList(); // If we only want the type as the value add just that to the list. if ($typeOnly) { foreach ($fields as $field) { $result[$field->Field] = preg_replace("/[(0-9)]/",'', $field->Type); } } // If we want the whole field data object add that to the list. else { foreach ($fields as $field) { $result[$field->Field] = $field; } } return $result; } /** * Get the details list of keys for a table. * * @param string $table The name of the table. * * @return array An arry of the column specification for the table. * * @since 11.1 * @throws DatabaseException */ public function getTableKeys($table) { // Get the details columns information. $this->setQuery( 'SHOW KEYS FROM '.$this->db->quoteName($table) ); $keys = $this->loadObjectList(); return $keys; } /** * Method to get an array of all tables in the database. * * @return array An array of all the tables in the database. * * @since 11.1 * @throws DatabaseException */ public function getTableList() { // Set the query to get the tables statement. $this->setQuery('SHOW TABLES'); $tables = $this->loadColumn(); return $tables; } /** * Get the version of the database connector. * * @return string The database connector version. * * @since 11.1 */ public function getVersion() { return mysqli_get_server_info($this->connection); } /** * Determines if the database engine supports UTF-8 character encoding. * * @return boolean True if supported. * * @since 11.1 */ public function hasUTF() { $verParts = explode('.', $this->getVersion()); return ($verParts[0] == 5 || ($verParts[0] == 4 && $verParts[1] == 1 && (int)$verParts[2] >= 2)); } /** * Method to get the auto-incremented value from the last INSERT statement. * * @return integer The value of the auto-increment field from the last inserted row. * * @since 11.1 */ public function insertid() { return mysqli_insert_id($this->connection); } /** * Execute the SQL statement. * * @return mixed A database cursor resource on success, boolean false on failure. * * @since 11.1 * @throws DatabaseException */ public function query() { if (!is_object($this->connection)) { // Legacy error handling switch based on the JError::$legacy switch. // @deprecated 12.1 if (JError::$legacy) { if ($this->debug) { JError::raiseError(500, 'JDatabaseMySQL::query: '.$this->errorNum.' - '.$this->errorMsg); } return false; } else { JLog::add(JText::sprintf('JLIB_DATABASE_QUERY_FAILED', $this->errorNum, $this->errorMsg), JLog::ERROR, 'database'); throw new DatabaseException; } } // Take a local copy so that we don't modify the original query and cause issues later $sql = $this->replacePrefix((string) $this->sql); if ($this->limit > 0 || $this->offset > 0) { $sql .= ' LIMIT '.$this->offset.', '.$this->limit; } // If debugging is enabled then let's log the query. if ($this->debug) { // Increment the query counter and add the query to the object queue. $this->count++; $this->log[] = $sql; JLog::add($sql, JLog::DEBUG, 'databasequery'); } // Reset the error values. $this->errorNum = 0; $this->errorMsg = ''; // Execute the query. $this->cursor = mysqli_query($this->connection, $sql); // If an error occurred handle it. if (!$this->cursor) { $this->errorNum = (int) mysqli_errno($this->connection); $this->errorMsg = (string) mysqli_error($this->connection).' SQL='.$sql; // Legacy error handling switch based on the JError::$legacy switch. // @deprecated 12.1 if (JError::$legacy) { if ($this->debug) { JError::raiseError(500, 'JDatabaseMySQL::query: '.$this->errorNum.' - '.$this->errorMsg); } return false; } else { JLog::add(JText::sprintf('JLIB_DATABASE_QUERY_FAILED', $this->errorNum, $this->errorMsg), JLog::ERROR, 'databasequery'); throw new DatabaseException; } } return $this->cursor; } /** * Select a database for use. * * @param string $database The name of the database to select for use. * * @return boolean True if the database was successfully selected. * * @since 11.1 * @throws DatabaseException */ public function select($database) { if (!$database) { return false; } if (!mysqli_select_db($this->connection, $database)) { // Legacy error handling switch based on the JError::$legacy switch. // @deprecated 12.1 if (JError::$legacy) { $this->errorNum = 3; $this->errorMsg = JText::_('JLIB_DATABASE_ERROR_DATABASE_CONNECT'); return false; } else { throw new DatabaseException(JText::_('JLIB_DATABASE_ERROR_DATABASE_CONNECT')); } } return true; } /** * Set the connection to use UTF-8 character encoding. * * @return bool True on success. * * @since 11.1 */ public function setUTF() { mysqli_query($this->connection, "SET NAMES 'utf8'"); } /** * Method to commit a transaction. * * @return void * * @since 11.1 * @throws DatabaseException */ public function transactionCommit() { $this->setQuery('COMMIT'); $this->query(); } /** * Method to roll back a transaction. * * @return void * * @since 11.1 * @throws DatabaseException */ public function transactionRollback() { $this->setQuery('ROLLBACK'); $this->query(); } /** * Method to initialize a transaction. * * @return void * * @since 11.1 * @throws DatabaseException */ public function transactionStart() { $this->setQuery('START TRANSACTION'); $this->query(); } /** * Method to fetch a row from the result set cursor as an array. * * @param mixed $cursor The optional result set cursor from which to fetch the row. * * @return mixed Either the next row from the result set or false if there are no more rows. * * @since 11.1 */ protected function fetchArray($cursor = null) { return mysqli_fetch_row($cursor ? $cursor : $this->cursor); } /** * Method to fetch a row from the result set cursor as an associative array. * * @param mixed $cursor The optional result set cursor from which to fetch the row. * * @return mixed Either the next row from the result set or false if there are no more rows. * * @since 11.1 */ protected function fetchAssoc($cursor = null) { return mysqli_fetch_assoc($cursor ? $cursor : $this->cursor); } /** * Method to fetch a row from the result set cursor as an object. * * @param mixed $cursor The optional result set cursor from which to fetch the row. * @param string $class The class name to use for the returned row object. * * @return mixed Either the next row from the result set or false if there are no more rows. * * @since 11.1 */ protected function fetchObject($cursor = null, $class = 'stdClass') { return mysqli_fetch_object($cursor ? $cursor : $this->cursor, $class); } /** * Method to free up the memory used for the result set. * * @param mixed $cursor The optional result set cursor from which to fetch the row. * * @return void * * @since 11.1 */ protected function freeResult($cursor = null) { mysqli_free_result($cursor ? $cursor : $this->cursor); } /** * Diagnostic method to return explain information for a query. * * @return string The explain output. * * @deprecated 12.1 * @since 11.1 */ public function explain() { // Deprecation warning. JLog::add('JDatabase::explain() is deprecated.', JLog::WARNING, 'deprecated'); // Backup the current query so we can reset it later. $backup = $this->sql; // Prepend the current query with EXPLAIN so we get the diagnostic data. $this->sql = 'EXPLAIN '.$this->sql; // Execute the query and get the result set cursor. if (!($cursor = $this->query())) { return null; } // Build the HTML table. $first = true; $buffer = '<table id="explain-sql">'; $buffer .= '<thead><tr><td colspan="99">'.$this->getQuery().'</td></tr>'; while ($row = $this->fetchAssoc($cursor)) { if ($first) { $buffer .= '<tr>'; foreach ($row as $k=>$v) { $buffer .= '<th>'.$k.'</th>'; } $buffer .= '</tr></thead><tbody>'; $first = false; } $buffer .= '<tr>'; foreach ($row as $k=>$v) { $buffer .= '<td>'.$v.'</td>'; } $buffer .= '</tr>'; } $buffer .= '</tbody></table>'; // Restore the original query to it's state before we ran the explain. $this->sql = $backup; // Free up system resources and return. $this->freeResult($cursor); return $buffer; } /** * Execute a query batch. * * @return mixed A database resource if successful, false if not. * * @deprecated 12.1 * @since 11.1 */ public function queryBatch($abortOnError=true, $transactionSafe = false) { // Deprecation warning. JLog::add('JDatabase::queryBatch() is deprecated.', JLog::WARNING, 'deprecated'); $sql = $this->replacePrefix((string) $this->sql); $this->errorNum = 0; $this->errorMsg = ''; // If the batch is meant to be transaction safe then we need to wrap it in a transaction. if ($transactionSafe) { $sql = 'START TRANSACTION;'.rtrim($sql, "; \t\r\n\0").'; COMMIT;'; } $queries = $this->splitSql($sql); $error = 0; foreach ($queries as $query) { $query = trim($query); if ($query != '') { $this->cursor = mysqli_query($this->connection, $query); if ($this->debug) { $this->count++; $this->log[] = $query; } if (!$this->cursor) { $error = 1; $this->errorNum .= mysqli_errno($this->connection) . ' '; $this->errorMsg .= mysqli_error($this->connection)." SQL=$query <br />"; if ($abortOnError) { return $this->cursor; } } } } return $error ? false : true; } }