Viewing file: Index.class.php (19.34 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?php /* vim: set expandtab sw=4 ts=4 sts=4: */ /** * holds the database index class * * @package phpMyAdmin */
/** * @since phpMyAdmin 3.0.0 * * @package phpMyAdmin */ class PMA_Index { /** * Class-wide storage container for indexes (caching, singleton) * * @var array */ protected static $_registry = array();
/** * @var string The name of the schema */ protected $_schema = '';
/** * @var string The name of the table */ protected $_table = '';
/** * @var string The name of the index */ protected $_name = '';
/** * Columns in index * * @var array */ protected $_columns = array();
/** * The index method used (BTREE, FULLTEXT, HASH, RTREE). * * @var string */ protected $_type = '';
/** * The index choice (PRIMARY, UNIQUE, INDEX, FULLTEXT) * * @var string */ protected $_choice = '';
/** * Various remarks. * * @var string */ protected $_remarks = '';
/** * Any comment provided for the index with a COMMENT attribute when the * index was created. * * @var string */ protected $_comment = '';
/** * @var integer 0 if the index cannot contain duplicates, 1 if it can. */ protected $_non_unique = 0;
/** * Indicates how the key is packed. NULL if it is not. * * @var string */ protected $_packed = null;
/** * Constructor * * @uses $this->set() * @param array $params */ public function __construct($params = array()) { $this->set($params); }
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 * * @uses PMA_Index::_loadIndexes() * @uses PMA_Index::$_registry * @param string $table * @param string $schema * @return array */ 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 * * @uses PMA_Index::_loadIndexes() * @uses PMA_Index::$_registry * @param string $table * @param string $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 * * @uses PMA_Index::$_registry * @uses PMA_DBI_fetch_result() * @uses PMA_backquote() * @uses PMA_Index * @uses PMA_Index->addColumn() * @param string $table * @param string $schema * @return boolean */ static protected function _loadIndexes($table, $schema) { if (isset(PMA_Index::$_registry[$schema][$table])) { return true; }
$_raw_indexes = PMA_DBI_fetch_result('SHOW INDEX FROM ' . PMA_backquote($schema) . '.' . PMA_backquote($table)); foreach ($_raw_indexes as $_each_index) { $_each_index['Schema'] = $schema; if (! isset(PMA_Index::$_registry[$schema][$table][$_each_index['Key_name']])) { $key = new PMA_Index($_each_index); PMA_Index::$_registry[$schema][$table][$_each_index['Key_name']] = $key; } else { $key = PMA_Index::$_registry[$schema][$table][$_each_index['Key_name']]; }
$key->addColumn($_each_index); }
return true; }
/** * Add column to index * * @uses $this->_columns * @uses PMA_Index_Column * @param array $params column params */ public function addColumn($params) { if (strlen($params['Column_name'])) { $this->_columns[$params['Column_name']] = new PMA_Index_Column($params); } }
public function addColumns($columns) { $_columns = array();
if (isset($columns['names'])) { // coming from form // $columns[names][] // $columns[sub_parts][] foreach ($columns['names'] as $key => $name) { $_columns[] = array( 'Column_name' => $name, 'Sub_part' => $columns['sub_parts'][$key], ); } } 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 * * @uses $this->_columns * @param string $column * @return boolean */ public function hasColumn($column) { return isset($this->_columns[$column]); }
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 ('0' == $this->_non_unique) { $this->_choice = 'UNIQUE'; } else { $this->_choice = 'INDEX'; } }
public function getColumnCount() { return count($this->_columns); }
public function getComment() { return $this->_comment; }
public function getRemarks() { return $this->_remarks; }
public function getComments() { $comments = $this->getRemarks(); if (strlen($comments)) { $comments .= "\n"; } $comments .= $this->getComment();
return $comments; }
public function getType() { return $this->_type; }
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', 'FULLTEXT', ); }
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 .= '<option value="' . $each_index_choice . '"' . (($this->_choice == $each_index_choice) ? ' selected="selected"' : '') . '>'. $each_index_choice . '</option>' . "\n"; }
return $html_options; }
public function getPacked() { return $this->_packed; }
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; }
public function getNonUnique() { return $this->_non_unique; }
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]; }
public function getName() { return $this->_name; }
public function setName($name) { $this->_name = (string) $name; }
public function getColumns() { return $this->_columns; }
/** * Show index data * * @param string $table The tablename * @param array $indexes_info Referenced info array * @param array $indexes_data Referenced data array * @param boolean $print_mode * @access public * @return array Index collection array */ static public function getView($table, $schema, $print_mode = false) { $indexes = PMA_Index::getFromTable($table, $schema);
if (count($indexes) < 1) { return PMA_Message::error(__('No index defined!'))->getDisplay(); }
$r = '';
$r .= '<h2>' . __('Indexes') . ': '; $r .= PMA_showMySQLDocu('optimization', 'optimizing-database-structure'); $r .= '</h2>'; $r .= '<table>'; $r .= '<thead>'; $r .= '<tr>'; if (! $print_mode) { $r .= '<th colspan="2">' . __('Action') . '</th>'; } $r .= '<th>' . __('Keyname') . '</th>'; $r .= '<th>' . __('Type') . '</th>'; $r .= '<th>' . __('Unique') . '</th>'; $r .= '<th>' . __('Packed') . '</th>'; $r .= '<th>' . __('Column') . '</th>'; $r .= '<th>' . __('Cardinality') . '</th>'; $r .= '<th>' . __('Collation') . '</th>'; $r .= '<th>' . __('Null') . '</th>'; $r .= '<th>' . __('Comment') . '</th>'; $r .= '</tr>'; $r .= '</thead>'; $r .= '<tbody>';
$odd_row = true; foreach ($indexes as $index) { $row_span = ' rowspan="' . $index->getColumnCount() . '" ';
$r .= '<tr class="' . ($odd_row ? 'odd' : 'even') . '">';
if (! $print_mode) { $this_params = $GLOBALS['url_params']; $this_params['index'] = $index->getName(); $r .= '<td ' . $row_span . '>' . ' <a href="tbl_indexes.php' . PMA_generate_common_url($this_params) . '">' . PMA_getIcon('b_edit.png', __('Edit')) . '</a>' . '</td>' . "\n";
$this_params = $GLOBALS['url_params']; if ($index->getName() == 'PRIMARY') { $this_params['sql_query'] = 'ALTER TABLE ' . PMA_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_backquote($table) . ' DROP INDEX ' . PMA_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 .= '<td ' . $row_span . '>'; $r .= '<input type="hidden" class="drop_primary_key_index_msg" value="' . $js_msg . '" />'; $r .= ' <a '; if ($GLOBALS['cfg']['AjaxEnable']) { $r .= 'class="drop_primary_key_index_anchor" '; } $r .= ' href="sql.php' . PMA_generate_common_url($this_params) . '" >' . PMA_getIcon('b_drop.png', __('Drop')) . '</a>' . '</td>' . "\n"; }
$r .= '<th ' . $row_span . '>' . htmlspecialchars($index->getName()) . '</th>'; $r .= '<td ' . $row_span . '>' . htmlspecialchars($index->getType()) . '</td>'; $r .= '<td ' . $row_span . '>' . $index->isUnique(true) . '</td>'; $r .= '<td ' . $row_span . '>' . $index->isPacked(true) . '</td>';
foreach ($index->getColumns() as $column) { if ($column->getSeqInIndex() > 1) { $r .= '<tr class="' . ($odd_row ? 'odd' : 'even') . '">'; } $r .= '<td>' . htmlspecialchars($column->getName()); if ($column->getSubPart()) { $r .= ' (' . $column->getSubPart() . ')'; } $r .= '</td>'; $r .= '<td>' . htmlspecialchars($column->getCardinality()) . '</td>'; $r .= '<td>' . htmlspecialchars($column->getCollation()) . '</td>'; $r .= '<td>' . htmlspecialchars($column->getNull()) . '</td>';
if ($column->getSeqInIndex() == 1) { $r .= '<td ' . $row_span . '>' . htmlspecialchars($index->getComments()) . '</td>'; } $r .= '</tr>'; } // end foreach $index['Sequences']
$odd_row = ! $odd_row; } // end while $r .= '</tbody>'; $r .= '</table>';
if (! $print_mode) { $r .= PMA_Index::findDuplicates($table, $schema); }
return $r; }
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 * * @uses is_string() * @uses is_array() * @uses count() * @uses array_pop() * @uses reset() * @uses current() * @access public * @param string name of table * @return string Output HTML */ 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::error(__('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; } }
/** * @package phpMyAdmin */ class PMA_Index_Column { /** * @var string The column name */ protected $_name = '';
/** * @var integer The column sequence number in the index, starting with 1. */ protected $_seq_in_index = 1;
/** * @var string How the column is sorted in the index. “A” (Ascending) or NULL (Not sorted) */ protected $_collation = null;
/** * The number of indexed characters if the column is only partly indexed, * NULL if the entire column is indexed. * * @var integer */ protected $_sub_part = null;
/** * Contains YES if the column may contain NULL. * If not, the column contains NO. * * @var string */ protected $_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 */ protected $_cardinality = 0;
public function __construct($params = array()) { $this->set($params); }
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']; } }
public function getName() { return $this->_name; }
public function getCollation() { return $this->_collation; }
public function getCardinality() { return $this->_cardinality; }
public function getNull() { return $this->_null; }
public function getSeqInIndex() { return $this->_seq_in_index; }
public function getSubPart() { return $this->_sub_part; }
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, ); } } ?>
|