mirror of
https://github.com/silverstripe/silverstripe-postgresql
synced 2024-10-22 17:05:45 +02:00
FIXED: Issue with correct extraction of index names from the database. The root cause of this issue was the way that columns from indxes were retrieved. It was assumed that the column names formed the index name, which isn't necessarily true (E.g. when the index is named "SearchFields"). The behaviour of the module was updated to create case-sensitive index and trigger names, which could then be used to later tell Silverstripe which indexes existed in the database. These could be compared to the SiteTree::$indexes property in a case-sensitive fashion to determine which indexes needed to be created / updated. This update fixes a lot of the unnecessary/broken DDL operations that occurred.
This commit is contained in:
parent
dc7334087c
commit
3291147c8e
@ -473,6 +473,38 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
return $tableName;
|
return $tableName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Builds the internal Postgres index name given the silverstripe table and index name
|
||||||
|
* @param string $tableName
|
||||||
|
* @param string $indexName
|
||||||
|
* @param string $prefix The optional prefix for the index. Defaults to "ix" for indexes.
|
||||||
|
* @return string The postgres name of the index
|
||||||
|
*/
|
||||||
|
function buildPostgresIndexName($tableName, $indexName, $prefix = 'ix') {
|
||||||
|
|
||||||
|
// Replace namespace character
|
||||||
|
$tableNameSafe = str_replace("\\", "_", $tableName);
|
||||||
|
|
||||||
|
// Assume all indexes also contain the table name
|
||||||
|
$indexNamePG = "{$prefix}_{$tableNameSafe}_{$indexName}";
|
||||||
|
|
||||||
|
// Limit to 63 characters
|
||||||
|
if (strlen($indexNamePG) > 63)
|
||||||
|
return substr($indexNamePG, 0, 63);
|
||||||
|
return $indexNamePG;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Builds the internal Postgres trigger name given the silverstripe table and trigger name
|
||||||
|
* @param string $tableName
|
||||||
|
* @param string $triggerName
|
||||||
|
* @return string The postgres name of the trigger
|
||||||
|
*/
|
||||||
|
function buildPostgresTriggerName($tableName, $triggerName) {
|
||||||
|
// Kind of cheating, but behaves the same way as indexes
|
||||||
|
return $this->buildPostgresIndexName($tableName, $triggerName, 'ts');
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Alter a table's schema.
|
* Alter a table's schema.
|
||||||
* @param $table The name of the table to alter
|
* @param $table The name of the table to alter
|
||||||
@ -488,10 +520,9 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
$alterList[] = "ADD \"$fieldName\" $fieldSpec";
|
$alterList[] = "ADD \"$fieldName\" $fieldSpec";
|
||||||
}
|
}
|
||||||
|
|
||||||
if($alteredFields) foreach($alteredFields as $indexName => $indexSpec) {
|
if ($alteredFields) foreach ($alteredFields as $indexName => $indexSpec) {
|
||||||
$val=$this->alterTableAlterColumn($tableName, $indexName, $indexSpec);
|
$val = $this->alterTableAlterColumn($tableName, $indexName, $indexSpec);
|
||||||
if($val!='')
|
if (!empty($val)) $alterList[] = $val;
|
||||||
$alterList[] = $val;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//Do we need to do anything with the tablespaces?
|
//Do we need to do anything with the tablespaces?
|
||||||
@ -511,6 +542,7 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
if($alteredIndexes) foreach($alteredIndexes as $indexName=>$indexSpec) {
|
if($alteredIndexes) foreach($alteredIndexes as $indexName=>$indexSpec) {
|
||||||
|
|
||||||
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
||||||
|
$indexNamePG = $this->buildPostgresIndexName($tableName, $indexName);
|
||||||
|
|
||||||
if($indexSpec['type']=='fulltext') {
|
if($indexSpec['type']=='fulltext') {
|
||||||
//For full text indexes, we need to drop the trigger, drop the index, AND drop the column
|
//For full text indexes, we need to drop the trigger, drop the index, AND drop the column
|
||||||
@ -526,13 +558,14 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// We'll execute these later:
|
// We'll execute these later:
|
||||||
$drop_triggers.= 'DROP TRIGGER IF EXISTS ts_' . strtolower($tableName) . '_' . strtolower($indexName) . ' ON "' . $tableName . '";';
|
$triggerNamePG = $this->buildPostgresTriggerName($tableName, $indexName);
|
||||||
$fulltexts.="ALTER TABLE \"{$tableName}\" ADD COLUMN {$ts_details['fulltexts']};";
|
$drop_triggers.= "DROP TRIGGER IF EXISTS \"$triggerNamePG\" ON \"$tableName\";";
|
||||||
|
$fulltexts .= "ALTER TABLE \"{$tableName}\" ADD COLUMN {$ts_details['fulltexts']};";
|
||||||
$triggers .= $ts_details['triggers'];
|
$triggers .= $ts_details['triggers'];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create index action (including fulltext)
|
// Create index action (including fulltext)
|
||||||
$alterIndexList[] = 'DROP INDEX IF EXISTS ix_' . strtolower($tableName) . '_' . strtolower($indexName) . ';';
|
$alterIndexList[] = "DROP INDEX IF EXISTS \"$indexNamePG\";";
|
||||||
$createIndex = $this->getIndexSqlDefinition($tableName, $indexName, $indexSpec);
|
$createIndex = $this->getIndexSqlDefinition($tableName, $indexName, $indexSpec);
|
||||||
if($createIndex!==false) $alterIndexList[] = $createIndex;
|
if($createIndex!==false) $alterIndexList[] = $createIndex;
|
||||||
}
|
}
|
||||||
@ -541,7 +574,7 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
if($newIndexes) foreach($newIndexes as $indexName => $indexSpec){
|
if($newIndexes) foreach($newIndexes as $indexName => $indexSpec){
|
||||||
|
|
||||||
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
||||||
|
$indexNamePG = $this->buildPostgresIndexName($tableName, $indexName);
|
||||||
//If we have a fulltext search request, then we need to create a special column
|
//If we have a fulltext search request, then we need to create a special column
|
||||||
//for GiST searches
|
//for GiST searches
|
||||||
//Pick up the new indexes here:
|
//Pick up the new indexes here:
|
||||||
@ -556,7 +589,7 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
//Check that this index doesn't already exist:
|
//Check that this index doesn't already exist:
|
||||||
$indexes=$this->indexList($tableName);
|
$indexes=$this->indexList($tableName);
|
||||||
if(isset($indexes[$indexName])){
|
if(isset($indexes[$indexName])){
|
||||||
$alterIndexList[] = 'DROP INDEX IF EXISTS ix_' . strtolower($tableName) . '_' . strtolower($indexName) . ';';
|
$alterIndexList[] = "DROP INDEX IF EXISTS \"$indexNamePG\";";
|
||||||
}
|
}
|
||||||
|
|
||||||
$createIndex=$this->getIndexSqlDefinition($tableName, $indexName, $indexSpec);
|
$createIndex=$this->getIndexSqlDefinition($tableName, $indexName, $indexSpec);
|
||||||
@ -609,8 +642,9 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//Lastly, clustering goes here:
|
//Lastly, clustering goes here:
|
||||||
if($advancedOptions && isset($advancedOptions['cluster'])){
|
if ($advancedOptions && isset($advancedOptions['cluster'])) {
|
||||||
DB::query("CLUSTER \"$tableName\" USING ix_{$tableName}_{$advancedOptions['cluster']};");
|
$clusterIndex = $this->buildPostgresIndexName($tableName, $advancedOptions['cluster']);
|
||||||
|
DB::query("CLUSTER \"$tableName\" USING \"$clusterIndex\";");
|
||||||
} else {
|
} else {
|
||||||
//Check that clustering is not on this table, and if it is, remove it:
|
//Check that clustering is not on this table, and if it is, remove it:
|
||||||
|
|
||||||
@ -939,9 +973,10 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Given an index specification in the form of a string ensure that each
|
* Given an index specification in the form of a string ensure that each
|
||||||
* column name is property quoted, stripping brackets
|
* column name is property quoted, stripping brackets and modifiers.
|
||||||
* @param string $spec The input specification
|
* This index may also be in the form of a "CREATE INDEX..." sql fragment
|
||||||
* @return string The properly quoted column list
|
* @param string $spec The input specification or query. E.g. 'unique (Column1, Column2)'
|
||||||
|
* @return string The properly quoted column list. E.g. '"Column1", "Column2"'
|
||||||
*/
|
*/
|
||||||
function quoteColumnSpecString($spec) {
|
function quoteColumnSpecString($spec) {
|
||||||
$bits = $this->explodeColumnString($spec);
|
$bits = $this->explodeColumnString($spec);
|
||||||
@ -1001,56 +1036,52 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
return $indexName;
|
return $indexName;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine index name and check for existence
|
// Determine index name
|
||||||
$tableCol= 'ix_' . str_replace("\\", "_", $tableName) . '_' . $indexName;
|
$tableCol = $this->buildPostgresIndexName($tableName, $indexName);
|
||||||
if(strlen($tableCol)>64){
|
|
||||||
$tableCol=substr($indexName, 0, 59) . rand(1000, 9999);
|
|
||||||
}
|
|
||||||
$existing=DB::query("SELECT tablename FROM pg_indexes WHERE indexname='" . strtolower($tableCol) . "';")->first();
|
|
||||||
if($existing) return false;
|
|
||||||
|
|
||||||
// Consolidate/Cleanup spec into array format
|
// Consolidate/Cleanup spec into array format
|
||||||
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
$indexSpec = $this->parseIndexSpec($indexName, $indexSpec);
|
||||||
|
|
||||||
//Misc options first:
|
//Misc options first:
|
||||||
$fillfactor=$where='';
|
$fillfactor = $where = '';
|
||||||
if(isset($indexSpec['fillfactor'])) {
|
if (isset($indexSpec['fillfactor'])) {
|
||||||
$fillfactor='WITH (FILLFACTOR = ' . $indexSpec['fillfactor'] . ')';
|
$fillfactor = 'WITH (FILLFACTOR = ' . $indexSpec['fillfactor'] . ')';
|
||||||
}
|
}
|
||||||
if(isset($indexSpec['where'])) {
|
if (isset($indexSpec['where'])) {
|
||||||
$where='WHERE ' . $indexSpec['where'];
|
$where = 'WHERE ' . $indexSpec['where'];
|
||||||
}
|
}
|
||||||
|
|
||||||
//create a type-specific index
|
//create a type-specific index
|
||||||
//NOTE: hash should be removed. This is only here to demonstrate how other indexes can be made
|
// NOTE: hash should be removed. This is only here to demonstrate how other indexes can be made
|
||||||
switch($indexSpec['type']){
|
// NOTE: Quote the index name to preserve case sensitivity
|
||||||
|
switch ($indexSpec['type']) {
|
||||||
case 'fulltext':
|
case 'fulltext':
|
||||||
// @see fulltext() for the definition of the trigger that ts_$IndexName uses for fulltext searching
|
// @see fulltext() for the definition of the trigger that ts_$IndexName uses for fulltext searching
|
||||||
$spec="create index $tableCol ON \"" . $tableName . "\" USING " . $this->default_fts_cluster_method . "(\"ts_" . $indexName . "\") $fillfactor $where";
|
$spec = "create index \"$tableCol\" ON \"$tableName\" USING " . $this->default_fts_cluster_method . "(\"ts_" . $indexName . "\") $fillfactor $where";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'unique':
|
case 'unique':
|
||||||
$spec="create unique index $tableCol ON \"" . $tableName . "\" (" . $indexSpec['value'] . ") $fillfactor $where";
|
$spec = "create unique index \"$tableCol\" ON \"$tableName\" (" . $indexSpec['value'] . ") $fillfactor $where";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'btree':
|
case 'btree':
|
||||||
$spec="create index $tableCol ON \"" . $tableName . "\" USING btree (" . $indexSpec['value'] . ") $fillfactor $where";
|
$spec = "create index \"$tableCol\" ON \"$tableName\" USING btree (" . $indexSpec['value'] . ") $fillfactor $where";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'hash':
|
case 'hash':
|
||||||
//NOTE: this is not a recommended index type
|
//NOTE: this is not a recommended index type
|
||||||
$spec="create index $tableCol ON \"" . $tableName . "\" USING hash (" . $indexSpec['value'] . ") $fillfactor $where";
|
$spec = "create index \"$tableCol\" ON \"$tableName\" USING hash (" . $indexSpec['value'] . ") $fillfactor $where";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'index':
|
case 'index':
|
||||||
//'index' is the same as default, just a normal index with the default type decided by the database.
|
//'index' is the same as default, just a normal index with the default type decided by the database.
|
||||||
default:
|
default:
|
||||||
$spec="create index $tableCol ON \"" . $tableName . "\" (" . $indexSpec['value'] . ") $fillfactor $where";
|
$spec = "create index \"$tableCol\" ON \"$tableName\" (" . $indexSpec['value'] . ") $fillfactor $where";
|
||||||
}
|
}
|
||||||
return trim($spec) . ';';
|
return trim($spec) . ';';
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDbSqlDefinition($tableName, $indexName, $indexSpec){
|
function getDbSqlDefinition($tableName, $indexName, $indexSpec) {
|
||||||
return $this->getIndexSqlDefinition($tableName, $indexName, $indexSpec, true);
|
return $this->getIndexSqlDefinition($tableName, $indexName, $indexSpec, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1072,10 +1103,40 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
$indexType = "index";
|
$indexType = "index";
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->query("DROP INDEX $indexName");
|
$this->query("DROP INDEX \"$indexName\"");
|
||||||
$this->query("ALTER TABLE \"$tableName\" ADD $indexType \"$indexName\" $indexFields");
|
$this->query("ALTER TABLE \"$tableName\" ADD $indexType \"$indexName\" $indexFields");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a trigger name attempt to determine the columns upon which it acts
|
||||||
|
* @param string $triggerName Postgres trigger name
|
||||||
|
* @return array List of columns
|
||||||
|
*/
|
||||||
|
protected function extractTriggerColumns($triggerName)
|
||||||
|
{
|
||||||
|
$trigger = DB::query($statement = sprintf(
|
||||||
|
"SELECT tgargs FROM pg_catalog.pg_trigger WHERE tgname='%s'", $this->addslashes($triggerName)
|
||||||
|
))->first();
|
||||||
|
|
||||||
|
// Trigger columns will be extracted in an ugly hex format with null-
|
||||||
|
// terminated strings, needs some coaxing into a readable format
|
||||||
|
$tgargsHex = $trigger['tgargs'];
|
||||||
|
$tgargs = array();
|
||||||
|
$tgarg = '';
|
||||||
|
for ($i = 0; $i < strlen($tgargsHex); $i+=2) {
|
||||||
|
$hexChar = substr($tgargsHex, $i, 2);
|
||||||
|
if($hexChar == '00') {
|
||||||
|
$tgargs[] = $tgarg;
|
||||||
|
$tgarg = '';
|
||||||
|
} else {
|
||||||
|
$tgarg .= chr(hexdec($hexChar));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Drop first two arguments (trigger name and config name) and implode into nice list
|
||||||
|
return array_slice($tgargs, 2);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the list of indexes in a table.
|
* Return the list of indexes in a table.
|
||||||
* @param string $table The table name.
|
* @param string $table The table name.
|
||||||
@ -1090,6 +1151,15 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
$indexList=Array();
|
$indexList=Array();
|
||||||
foreach($indexes as $index) {
|
foreach($indexes as $index) {
|
||||||
|
|
||||||
|
// Determine the name of the index
|
||||||
|
if (stristr($index['indexname'], '_pkey')) {
|
||||||
|
$indexName = 'ID';
|
||||||
|
} else {
|
||||||
|
// Extract index name by splitting the ix_TableName_ from the start of the name
|
||||||
|
$indexNamePrefix = $this->buildPostgresIndexName($table, '');
|
||||||
|
$indexName = substr($index['indexname'], strlen($indexNamePrefix));
|
||||||
|
}
|
||||||
|
|
||||||
//We don't actually need the entire created command, just a few bits:
|
//We don't actually need the entire created command, just a few bits:
|
||||||
$prefix='';
|
$prefix='';
|
||||||
|
|
||||||
@ -1111,19 +1181,19 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
$prefix='using rtree ';
|
$prefix='using rtree ';
|
||||||
}
|
}
|
||||||
|
|
||||||
$value=explode(' ', substr($index['indexdef'], strpos($index['indexdef'], ' USING ')+7));
|
// For fulltext indexes we need to extract the columns from another source
|
||||||
|
if (stristr($index['indexdef'], 'using gin')) {
|
||||||
if(sizeof($value)>2){
|
$prefix = 'fulltext ';
|
||||||
for($i=2; $i<sizeof($value); $i++) {
|
// Extract trigger information from postgres
|
||||||
$value[1].=$value[$i];
|
$triggerName = $this->buildPostgresTriggerName($table, $indexName);
|
||||||
}
|
$columns = $this->extractTriggerColumns($triggerName);
|
||||||
|
$columnString = $this->implodeColumnList($columns);
|
||||||
|
} else {
|
||||||
|
$columnString = $this->quoteColumnSpecString($index['indexdef']);
|
||||||
}
|
}
|
||||||
|
|
||||||
$key=substr($value[1], 0, strpos($value[1], ')'));
|
$indexList[$indexName]['indexname'] = $index['indexname'];
|
||||||
$key=trim(trim(str_replace("\"", '', $key), '()'));
|
$indexList[$indexName]['spec'] = "$prefix($columnString)";
|
||||||
$indexList[$key]['indexname']=$index['indexname'];
|
|
||||||
$indexList[$key]['spec']=$prefix . '("' . preg_replace('/ *, */','","',$key) . '")';
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return isset($indexList) ? $indexList : null;
|
return isset($indexList) ? $indexList : null;
|
||||||
@ -1479,16 +1549,16 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
//For full text search, we need to create a column for the index
|
//For full text search, we need to create a column for the index
|
||||||
$columns = $this->quoteColumnSpecString($this_index['value']);
|
$columns = $this->quoteColumnSpecString($this_index['value']);
|
||||||
|
|
||||||
$fulltexts="\"ts_$name\" tsvector";
|
$fulltexts = "\"ts_$name\" tsvector";
|
||||||
$triggerName="ts_{$tableName}_{$name}";
|
$triggerName = $this->buildPostgresTriggerName($tableName, $name);
|
||||||
$language=$this->get_search_language();
|
$language = $this->get_search_language();
|
||||||
|
|
||||||
$this->dropTrigger($triggerName, $tableName);
|
$this->dropTrigger($triggerName, $tableName);
|
||||||
$triggers="CREATE TRIGGER $triggerName BEFORE INSERT OR UPDATE
|
$triggers = "CREATE TRIGGER \"$triggerName\" BEFORE INSERT OR UPDATE
|
||||||
ON \"$tableName\" FOR EACH ROW EXECUTE PROCEDURE
|
ON \"$tableName\" FOR EACH ROW EXECUTE PROCEDURE
|
||||||
tsvector_update_trigger(\"ts_$name\", 'pg_catalog.$language', $columns);";
|
tsvector_update_trigger(\"ts_$name\", 'pg_catalog.$language', $columns);";
|
||||||
|
|
||||||
return Array('name'=>$name, 'ts_name'=>"ts_{$name}", 'fulltexts'=>$fulltexts, 'triggers'=>$triggers);
|
return Array('name' => $name, 'ts_name' => "ts_{$name}", 'fulltexts' => $fulltexts, 'triggers' => $triggers);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1594,13 +1664,9 @@ class PostgreSQLDatabase extends SS_Database {
|
|||||||
/*
|
/*
|
||||||
* This changes the index name depending on database requirements.
|
* This changes the index name depending on database requirements.
|
||||||
*/
|
*/
|
||||||
function modifyIndex($index, $spec){
|
|
||||||
|
|
||||||
if(is_array($spec) && $spec['type']=='fulltext')
|
|
||||||
return 'ts_' . str_replace(',', '_', $index);
|
|
||||||
else
|
|
||||||
return str_replace('_', ',', $index);
|
|
||||||
|
|
||||||
|
function modifyIndex($index, $spec) {
|
||||||
|
return $index;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user