set($params); } /** * Creates(if not already created) and returns the corresponding Index object * * @param string $schema database name * @param string $table table name * @param string $index_name index name * * @return PMA_Index corresponding Index object */ static public function singleton($schema, $table, $index_name = '') { PMA_Index::_loadIndexes($table, $schema); if (! isset(PMA_Index::$_registry[$schema][$table][$index_name])) { $index = new PMA_Index; if (strlen($index_name)) { $index->setName($index_name); PMA_Index::$_registry[$schema][$table][$index->getName()] = $index; } return $index; } else { return PMA_Index::$_registry[$schema][$table][$index_name]; } } /** * returns an array with all indexes from the given table * * @param string $table table * @param string $schema schema * * @return array array of indexes */ static public function getFromTable($table, $schema) { PMA_Index::_loadIndexes($table, $schema); if (isset(PMA_Index::$_registry[$schema][$table])) { return PMA_Index::$_registry[$schema][$table]; } else { return array(); } } /** * return primary if set, false otherwise * * @param string $table table * @param string $schema schema * * @return mixed primary index or false if no one exists */ static public function getPrimary($table, $schema) { PMA_Index::_loadIndexes($table, $schema); if (isset(PMA_Index::$_registry[$schema][$table]['PRIMARY'])) { return PMA_Index::$_registry[$schema][$table]['PRIMARY']; } else { return false; } } /** * Load index data for table * * @param string $table table * @param string $schema schema * * @return boolean whether loading was successful */ static private function _loadIndexes($table, $schema) { if (isset(PMA_Index::$_registry[$schema][$table])) { return true; } $_raw_indexes = $GLOBALS['dbi']->getTableIndexes($schema, $table); foreach ($_raw_indexes as $_each_index) { $_each_index['Schema'] = $schema; $keyName = $_each_index['Key_name']; if (! isset(PMA_Index::$_registry[$schema][$table][$keyName])) { $key = new PMA_Index($_each_index); PMA_Index::$_registry[$schema][$table][$keyName] = $key; } else { $key = PMA_Index::$_registry[$schema][$table][$keyName]; } $key->addColumn($_each_index); } return true; } /** * Add column to index * * @param array $params column params * * @return void */ public function addColumn($params) { if (strlen($params['Column_name'])) { $this->_columns[$params['Column_name']] = new PMA_Index_Column($params); } } /** * Adds a list of columns to the index * * @param array $columns array containing details about the columns * * @return void */ public function addColumns($columns) { $_columns = array(); if (isset($columns['names'])) { // coming from form // $columns[names][] // $columns[sub_parts][] foreach ($columns['names'] as $key => $name) { $sub_part = isset($columns['sub_parts'][$key]) ? $columns['sub_parts'][$key] : ''; $_columns[] = array( 'Column_name' => $name, 'Sub_part' => $sub_part, ); } } else { // coming from SHOW INDEXES // $columns[][name] // $columns[][sub_part] // ... $_columns = $columns; } foreach ($_columns as $column) { $this->addColumn($column); } } /** * Returns true if $column indexed in this index * * @param string $column the column * * @return boolean true if $column indexed in this index */ public function hasColumn($column) { return isset($this->_columns[$column]); } /** * Sets index details * * @param array $params index details * * @return void */ public function set($params) { if (isset($params['columns'])) { $this->addColumns($params['columns']); } if (isset($params['Schema'])) { $this->_schema = $params['Schema']; } if (isset($params['Table'])) { $this->_table = $params['Table']; } if (isset($params['Key_name'])) { $this->_name = $params['Key_name']; } if (isset($params['Index_type'])) { $this->_type = $params['Index_type']; } if (isset($params['Comment'])) { $this->_remarks = $params['Comment']; } if (isset($params['Index_comment'])) { $this->_comment = $params['Index_comment']; } if (isset($params['Non_unique'])) { $this->_non_unique = $params['Non_unique']; } if (isset($params['Packed'])) { $this->_packed = $params['Packed']; } if ('PRIMARY' == $this->_name) { $this->_choice = 'PRIMARY'; } elseif ('FULLTEXT' == $this->_type) { $this->_choice = 'FULLTEXT'; } elseif ('SPATIAL' == $this->_type) { $this->_choice = 'SPATIAL'; } elseif ('0' == $this->_non_unique) { $this->_choice = 'UNIQUE'; } else { $this->_choice = 'INDEX'; } } /** * Returns the number of columns of the index * * @return integer the number of the columns */ public function getColumnCount() { return count($this->_columns); } /** * Returns the index comment * * @return string index comment */ public function getComment() { return $this->_comment; } /** * Returns index remarks * * @return string index remarks */ public function getRemarks() { return $this->_remarks; } /** * Returns concatenated remarks and comment * * @return string concatenated remarks and comment */ public function getComments() { $comments = $this->getRemarks(); if (strlen($comments)) { $comments .= "\n"; } $comments .= $this->getComment(); return $comments; } /** * Returns index type ((BTREE, SPATIAL, FULLTEXT, HASH, RTREE) * * @return string index type */ public function getType() { return $this->_type; } /** * Returns index choice (PRIMARY, UNIQUE, INDEX, SPATIAL, FULLTEXT) * * @return string index choice */ public function getChoice() { return $this->_choice; } /** * Return a list of all index choices * * @return array index choices */ static public function getIndexChoices() { return array( 'PRIMARY', 'INDEX', 'UNIQUE', 'SPATIAL', 'FULLTEXT', ); } /** * Returns HTML for the index choice selector * * @return string HTML for the index choice selector */ public function generateIndexSelector() { $html_options = ''; foreach (PMA_Index::getIndexChoices() as $each_index_choice) { if ($each_index_choice === 'PRIMARY' && $this->_choice !== 'PRIMARY' && PMA_Index::getPrimary($this->_table, $this->_schema) ) { // skip PRIMARY if there is already one in the table continue; } $html_options .= '' . "\n"; } return $html_options; } /** * Returns how the index is packed * * @return string how the index is packed */ public function getPacked() { return $this->_packed; } /** * Returns 'No'/false if the index is not packed, * how the index is packed if packed * * @param boolean $as_text whether to output should be in text * * @return mixed how index is paked */ public function isPacked($as_text = false) { if ($as_text) { $r = array( '0' => __('No'), '1' => __('Yes'), ); } else { $r = array( '0' => false, '1' => true, ); } if (null === $this->_packed) { return $r[0]; } return $this->_packed; } /** * Returns integer 0 if the index cannot contain duplicates, 1 if it can * * @return integer 0 if the index cannot contain duplicates, 1 if it can */ public function getNonUnique() { return $this->_non_unique; } /** * Returns whether the index is a 'Unique' index * * @param boolean $as_text whether to output should be in text * * @return mixed whether the index is a 'Unique' index */ public function isUnique($as_text = false) { if ($as_text) { $r = array( '0' => __('Yes'), '1' => __('No'), ); } else { $r = array( '0' => true, '1' => false, ); } return $r[$this->_non_unique]; } /** * Returns the name of the index * * @return string the name of the index */ public function getName() { return $this->_name; } /** * Sets the name of the index * * @param string $name index name * * @return void */ public function setName($name) { $this->_name = (string) $name; } /** * Returns the columns of the index * * @return array the columns of the index */ public function getColumns() { return $this->_columns; } /** * Show index data * * @param string $table The table name * @param string $schema The schema name * @param boolean $print_mode Whether the output is for the print mode * * @return array Index collection array * * @access public */ static public function getView($table, $schema, $print_mode = false) { $indexes = PMA_Index::getFromTable($table, $schema); $no_indexes_class = count($indexes) > 0 ? ' hide' : ''; $no_indexes = "
"; $no_indexes .= PMA_Message::notice(__('No index defined!'))->getDisplay(); $no_indexes .= '
'; if (! $print_mode) { $r = '
'; $r .= '' . __('Indexes'); $r .= PMA_Util::showMySQLDocu('optimizing-database-structure'); $r .= ''; $r .= $no_indexes; if (count($indexes) < 1) { $r .= '
'; return $r; } $r .= PMA_Index::findDuplicates($table, $schema); } else { $r = '

' . __('Indexes') . '

'; $r .= $no_indexes; if (count($indexes) < 1) { return $r; } } $r .= ''; $r .= ''; $r .= ''; if (! $print_mode) { $r .= ''; } $r .= ''; $r .= ''; $r .= ''; $r .= ''; $r .= ''; $r .= ''; $r .= ''; $r .= ''; if (PMA_MYSQL_INT_VERSION > 50500) { $r .= ''; } $r .= ''; $r .= ''; $r .= ''; $odd_row = true; foreach ($indexes as $index) { $row_span = ' rowspan="' . $index->getColumnCount() . '" '; $r .= ''; if (! $print_mode) { $this_params = $GLOBALS['url_params']; $this_params['index'] = $index->getName(); $r .= '' . "\n"; $this_params = $GLOBALS['url_params']; if ($index->getName() == 'PRIMARY') { $this_params['sql_query'] = 'ALTER TABLE ' . PMA_Util::backquote($table) . ' DROP PRIMARY KEY;'; $this_params['message_to_show'] = __('The primary key has been dropped'); $js_msg = PMA_jsFormat( 'ALTER TABLE ' . $table . ' DROP PRIMARY KEY' ); } else { $this_params['sql_query'] = 'ALTER TABLE ' . PMA_Util::backquote($table) . ' DROP INDEX ' . PMA_Util::backquote($index->getName()) . ';'; $this_params['message_to_show'] = sprintf( __('Index %s has been dropped.'), $index->getName() ); $js_msg = PMA_jsFormat( 'ALTER TABLE ' . $table . ' DROP INDEX ' . $index->getName() . ';' ); } $r .= '' . "\n"; } if (! $print_mode) { $r .= ''; } else { $r .= ''; } $r .= ''; $r .= ''; $r .= ''; foreach ($index->getColumns() as $column) { if ($column->getSeqInIndex() > 1) { $r .= ''; } $r .= ''; $r .= ''; $r .= ''; $r .= ''; if (PMA_MYSQL_INT_VERSION > 50500 && $column->getSeqInIndex() == 1 ) { $r .= ''; } $r .= ''; } // end foreach $index['Sequences'] $odd_row = ! $odd_row; } // end while $r .= ''; $r .= '
' . __('Action') . '' . __('Keyname') . '' . __('Type') . '' . __('Unique') . '' . __('Packed') . '' . __('Column') . '' . __('Cardinality') . '' . __('Collation') . '' . __('Null') . '' . __('Comment') . '
' . ' ' . PMA_Util::getIcon('b_edit.png', __('Edit')) . '' . ''; $r .= ''; $r .= ' ' . PMA_Util::getIcon('b_drop.png', __('Drop')) . '' . '' . htmlspecialchars($index->getName()) . '' . htmlspecialchars($index->getName()) . '' . htmlspecialchars($index->getType()) . '' . $index->isUnique(true) . '' . $index->isPacked(true) . '
' . htmlspecialchars($column->getName()); if ($column->getSubPart()) { $r .= ' (' . $column->getSubPart() . ')'; } $r .= '' . htmlspecialchars($column->getCardinality()) . '' . htmlspecialchars($column->getCollation()) . '' . htmlspecialchars($column->getNull(true)) . '' . htmlspecialchars($index->getComments()) . '
'; if (! $print_mode) { $r .= ''; } return $r; } /** * Gets the properties in an array for comparison purposes * * @return array an array containing the properties of the index */ public function getCompareData() { $data = array( // 'Non_unique' => $this->_non_unique, 'Packed' => $this->_packed, 'Index_type' => $this->_type, ); foreach ($this->_columns as $column) { $data['columns'][] = $column->getCompareData(); } return $data; } /** * Function to check over array of indexes and look for common problems * * @param string $table table name * @param string $schema schema name * * @return string Output HTML * @access public */ static public function findDuplicates($table, $schema) { $indexes = PMA_Index::getFromTable($table, $schema); $output = ''; // count($indexes) < 2: // there is no need to check if there less than two indexes if (count($indexes) < 2) { return $output; } // remove last index from stack and ... while ($while_index = array_pop($indexes)) { // ... compare with every remaining index in stack foreach ($indexes as $each_index) { if ($each_index->getCompareData() !== $while_index->getCompareData()) { continue; } // did not find any difference // so it makes no sense to have this two equal indexes $message = PMA_Message::notice( __('The indexes %1$s and %2$s seem to be equal and one of them could possibly be removed.') ); $message->addParam($each_index->getName()); $message->addParam($while_index->getName()); $output .= $message->getDisplay(); // there is no need to check any further indexes if we have already // found that this one has a duplicate continue 2; } } return $output; } } /** * Index column wrapper * * @package PhpMyAdmin */ class PMA_Index_Column { /** * @var string The column name */ private $_name = ''; /** * @var integer The column sequence number in the index, starting with 1. */ private $_seq_in_index = 1; /** * @var string How the column is sorted in the index. “A” (Ascending) or * NULL (Not sorted) */ private $_collation = null; /** * The number of indexed characters if the column is only partly indexed, * NULL if the entire column is indexed. * * @var integer */ private $_sub_part = null; /** * Contains YES if the column may contain NULL. * If not, the column contains NO. * * @var string */ private $_null = ''; /** * An estimate of the number of unique values in the index. This is updated * by running ANALYZE TABLE or myisamchk -a. Cardinality is counted based on * statistics stored as integers, so the value is not necessarily exact even * for small tables. The higher the cardinality, the greater the chance that * MySQL uses the index when doing joins. * * @var integer */ private $_cardinality = null; /** * Constructor * * @param array $params an array containing the parameters of the index column */ public function __construct($params = array()) { $this->set($params); } /** * Sets parameters of the index column * * @param array $params an array containing the parameters of the index column * * @return void */ public function set($params) { if (isset($params['Column_name'])) { $this->_name = $params['Column_name']; } if (isset($params['Seq_in_index'])) { $this->_seq_in_index = $params['Seq_in_index']; } if (isset($params['Collation'])) { $this->_collation = $params['Collation']; } if (isset($params['Cardinality'])) { $this->_cardinality = $params['Cardinality']; } if (isset($params['Sub_part'])) { $this->_sub_part = $params['Sub_part']; } if (isset($params['Null'])) { $this->_null = $params['Null']; } } /** * Returns the column name * * @return string column name */ public function getName() { return $this->_name; } /** * Return the column collation * * @return string column collation */ public function getCollation() { return $this->_collation; } /** * Returns the cardinality of the column * * @return int cardinality of the column */ public function getCardinality() { return $this->_cardinality; } /** * Returns whether the column is nullable * * @param boolean $as_text whether to returned the string representation * * @return mixed nullability of the column. True/false or Yes/No depending * on the value of the $as_text parameter */ public function getNull($as_text = false) { return $as_text ? (!$this->_null || $this->_null == 'NO' ? __('No') : __('Yes')) : $this->_null; } /** * Returns the sequence number of the column in the index * * @return int sequence number of the column in the index */ public function getSeqInIndex() { return $this->_seq_in_index; } /** * Returns the number of indexed characters if the column is only * partly indexed * * @return int the number of indexed characters */ public function getSubPart() { return $this->_sub_part; } /** * Gets the properties in an array for comparison purposes * * @return array an array containing the properties of the index column */ public function getCompareData() { return array( 'Column_name' => $this->_name, 'Seq_in_index' => $this->_seq_in_index, 'Collation' => $this->_collation, 'Sub_part' => $this->_sub_part, 'Null' => $this->_null, ); } } ?>