2013-04-03 17:27:11 +13:00
< ? php
2016-07-04 12:44:09 +12:00
namespace SilverStripe\PostgreSQL ;
2016-09-01 16:22:17 +12:00
use SilverStripe\Dev\Deprecation ;
2016-07-04 12:44:09 +12:00
use SilverStripe\ORM\Connect\DBSchemaManager ;
use SilverStripe\ORM\DB ;
2013-04-03 17:27:11 +13:00
/**
* PostgreSQL schema manager
2016-07-04 12:44:09 +12:00
*
2013-04-03 17:27:11 +13:00
* @ package sapphire
* @ subpackage model
*/
2015-12-18 07:18:01 +13:00
class PostgreSQLSchemaManager extends DBSchemaManager
{
2016-01-21 09:49:28 +13:00
/**
2016-07-04 12:44:09 +12:00
* Identifier for this schema , used for configuring schema - specific table
2016-01-21 09:49:28 +13:00
* creation options
*/
const ID = 'PostgreSQL' ;
/**
* Instance of the database controller this schema belongs to
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ var PostgreSQLDatabase
*/
protected $database = null ;
/**
* This holds a copy of all the constraint results that are returned
* via the function constraintExists () . This is a bit faster than
* repeatedly querying this column , and should allow the database
* to use it ' s built - in caching features for better queries .
*
* @ var array
*/
protected static $cached_constraints = array ();
/**
*
* This holds a copy of all the queries that run through the function fieldList ()
* This is one of the most - often called functions , and repeats itself a great deal in the unit tests .
*
* @ var array
*/
protected static $cached_fieldlists = array ();
2015-12-18 07:18:01 +13:00
protected function indexKey ( $table , $index , $spec )
{
2016-01-21 09:49:28 +13:00
return $this -> buildPostgresIndexName ( $table , $index );
}
/**
* Creates a postgres database , ignoring model_schema_as_database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
*/
2015-12-18 07:18:01 +13:00
public function createPostgresDatabase ( $name )
{
2016-01-21 09:49:28 +13:00
$this -> query ( " CREATE DATABASE \" $name\ " ; " );
}
2015-12-18 07:18:01 +13:00
public function createDatabase ( $name )
{
2016-01-21 09:49:28 +13:00
if ( PostgreSQLDatabase :: model_schema_as_database ()) {
$schemaName = $this -> database -> databaseToSchemaName ( $name );
return $this -> createSchema ( $schemaName );
}
return $this -> createPostgresDatabase ( $name );
}
/**
* Determines if a postgres database exists , ignoring model_schema_as_database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
* @ return boolean
*/
2015-12-18 07:18:01 +13:00
public function postgresDatabaseExists ( $name )
{
2016-01-21 09:49:28 +13:00
$result = $this -> preparedQuery ( " SELECT datname FROM pg_database WHERE datname = ?; " , array ( $name ));
return $result -> first () ? true : false ;
}
2015-12-18 07:18:01 +13:00
public function databaseExists ( $name )
{
2016-01-21 09:49:28 +13:00
if ( PostgreSQLDatabase :: model_schema_as_database ()) {
$schemaName = $this -> database -> databaseToSchemaName ( $name );
return $this -> schemaExists ( $schemaName );
}
return $this -> postgresDatabaseExists ( $name );
}
/**
* Determines the list of all postgres databases , ignoring model_schema_as_database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ return array
*/
2015-12-18 07:18:01 +13:00
public function postgresDatabaseList ()
{
2016-01-21 09:49:28 +13:00
return $this -> query ( " SELECT datname FROM pg_database WHERE datistemplate=false; " ) -> column ();
}
2015-12-18 07:18:01 +13:00
public function databaseList ()
{
2016-01-21 09:49:28 +13:00
if ( PostgreSQLDatabase :: model_schema_as_database ()) {
$schemas = $this -> schemaList ();
$names = array ();
foreach ( $schemas as $schema ) {
$names [] = $this -> database -> schemaToDatabaseName ( $schema );
}
return array_unique ( $names );
}
return $this -> postgresDatabaseList ();
}
2017-11-17 10:58:14 +13:00
2016-01-21 09:49:28 +13:00
/**
* Drops a postgres database , ignoring model_schema_as_database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
*/
2015-12-18 07:18:01 +13:00
public function dropPostgresDatabase ( $name )
{
2016-01-21 09:49:28 +13:00
$nameSQL = $this -> database -> escapeIdentifier ( $name );
$this -> query ( " DROP DATABASE $nameSQL ; " );
}
2015-12-18 07:18:01 +13:00
public function dropDatabase ( $name )
{
2016-01-21 09:49:28 +13:00
if ( PostgreSQLDatabase :: model_schema_as_database ()) {
$schemaName = $this -> database -> databaseToSchemaName ( $name );
2016-07-04 12:44:09 +12:00
$this -> dropSchema ( $schemaName );
return ;
2016-01-21 09:49:28 +13:00
}
$this -> dropPostgresDatabase ( $name );
}
/**
* Returns true if the schema exists in the current database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
* @ return boolean
*/
2015-12-18 07:18:01 +13:00
public function schemaExists ( $name )
{
2016-01-21 09:49:28 +13:00
return $this -> preparedQuery (
" SELECT nspname FROM pg_catalog.pg_namespace WHERE nspname = ?; " ,
array ( $name )
) -> first () ? true : false ;
}
/**
* Creates a schema in the current database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
*/
2015-12-18 07:18:01 +13:00
public function createSchema ( $name )
{
2016-01-21 09:49:28 +13:00
$nameSQL = $this -> database -> escapeIdentifier ( $name );
$this -> query ( " CREATE SCHEMA $nameSQL ; " );
}
/**
* Drops a schema from the database . Use carefully !
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
*/
2015-12-18 07:18:01 +13:00
public function dropSchema ( $name )
{
2016-01-21 09:49:28 +13:00
$nameSQL = $this -> database -> escapeIdentifier ( $name );
$this -> query ( " DROP SCHEMA $nameSQL CASCADE; " );
}
/**
* Returns the list of all available schemas on the current database
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ return array
*/
2015-12-18 07:18:01 +13:00
public function schemaList ()
{
2017-11-17 10:58:14 +13:00
return $this -> query (
" SELECT nspname
2016-01-21 09:28:37 +13:00
FROM pg_catalog . pg_namespace
WHERE nspname <> 'information_schema' AND nspname !~ E '^pg_' "
2016-01-21 09:49:28 +13:00
) -> column ();
}
2015-12-18 07:18:01 +13:00
public function createTable ( $table , $fields = null , $indexes = null , $options = null , $advancedOptions = null )
{
2017-11-17 10:58:14 +13:00
$fieldSchemas = " " ;
2015-12-18 07:18:01 +13:00
if ( $fields ) {
foreach ( $fields as $k => $v ) {
2016-01-21 09:49:28 +13:00
$fieldSchemas .= " \" $k\ " $v , \n " ;
}
}
if ( ! empty ( $options [ self :: ID ])) {
$addOptions = $options [ self :: ID ];
} else {
$addOptions = null ;
}
//First of all, does this table already exist
$doesExist = $this -> hasTable ( $table );
if ( $doesExist ) {
// Table already exists, just return the name, in line with baseclass documentation.
return $table ;
}
//If we have a fulltext search request, then we need to create a special column
//for GiST searches
$fulltexts = '' ;
2017-11-17 10:58:14 +13:00
$triggers = [];
2016-01-21 09:49:28 +13:00
if ( $indexes ) {
foreach ( $indexes as $name => $this_index ) {
if ( is_array ( $this_index ) && $this_index [ 'type' ] == 'fulltext' ) {
$ts_details = $this -> fulltext ( $this_index , $table , $name );
$fulltexts .= $ts_details [ 'fulltexts' ] . ', ' ;
2017-11-17 10:58:14 +13:00
$triggers [] = $ts_details [ 'triggers' ];
2016-01-21 09:49:28 +13:00
}
}
}
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
$indexQueries = [];
2015-12-18 07:18:01 +13:00
if ( $indexes ) {
foreach ( $indexes as $k => $v ) {
2017-11-17 10:58:14 +13:00
$indexQueries [] = $this -> getIndexSqlDefinition ( $table , $k , $v );
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
}
2016-01-21 09:49:28 +13:00
//Do we need to create a tablespace for this item?
if ( $advancedOptions && isset ( $advancedOptions [ 'tablespace' ])) {
$this -> createOrReplaceTablespace (
$advancedOptions [ 'tablespace' ][ 'name' ],
$advancedOptions [ 'tablespace' ][ 'location' ]
);
$tableSpace = ' TABLESPACE ' . $advancedOptions [ 'tablespace' ][ 'name' ];
2015-12-18 07:18:01 +13:00
} else {
2016-01-21 09:49:28 +13:00
$tableSpace = '' ;
2015-12-18 07:18:01 +13:00
}
2017-11-17 10:58:14 +13:00
$this -> query (
" CREATE TABLE \" $table\ " (
$fieldSchemas
$fulltexts
primary key ( \ " ID \" )
) $tableSpace $addOptions "
);
foreach ( $indexQueries as $indexQuery ) {
$this -> query ( $indexQuery );
}
2013-04-03 17:27:11 +13:00
2017-11-17 10:58:14 +13:00
foreach ( $triggers as $trigger ) {
$this -> query ( $trigger );
2016-01-21 09:49:28 +13:00
}
//If we have a partitioning requirement, we do that here:
if ( $advancedOptions && isset ( $advancedOptions [ 'partitions' ])) {
$this -> createOrReplacePartition ( $table , $advancedOptions [ 'partitions' ], $indexes , $advancedOptions );
}
//Lastly, clustering goes here:
if ( $advancedOptions && isset ( $advancedOptions [ 'cluster' ])) {
2017-11-17 10:58:14 +13:00
$this -> query ( " CLUSTER \" $table\ " USING \ " { $advancedOptions [ 'cluster' ] } \" " );
2016-01-21 09:49:28 +13:00
}
return $table ;
}
/**
* Builds the internal Postgres index name given the silverstripe table and index name
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName
2016-07-04 12:44:09 +12:00
* @ param string $indexName
2016-01-21 09:49:28 +13:00
* @ param string $prefix The optional prefix for the index . Defaults to " ix " for indexes .
* @ return string The postgres name of the index
*/
2015-12-18 07:18:01 +13:00
protected function buildPostgresIndexName ( $tableName , $indexName , $prefix = 'ix' )
{
2016-01-21 09:49:28 +13:00
// Assume all indexes also contain the table name
// MD5 the table/index name combo to keep it to a fixed length.
// Exclude the prefix so that the trigger name can be easily generated from the index name
$indexNamePG = " { $prefix } _ " . md5 ( " { $tableName } _ { $indexName } " );
// Limit to 63 characters
if ( strlen ( $indexNamePG ) > 63 ) {
return substr ( $indexNamePG , 0 , 63 );
} else {
return $indexNamePG ;
}
}
/**
* Builds the internal Postgres trigger name given the silverstripe table and trigger name
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName
* @ param string $triggerName
* @ return string The postgres name of the trigger
*/
2015-12-18 07:18:01 +13:00
public function buildPostgresTriggerName ( $tableName , $triggerName )
{
2016-01-21 09:49:28 +13:00
// Kind of cheating, but behaves the same way as indexes
return $this -> buildPostgresIndexName ( $tableName , $triggerName , 'ts' );
}
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
public function alterTable (
$table ,
$newFields = null ,
$newIndexes = null ,
$alteredFields = null ,
$alteredIndexes = null ,
$alteredOptions = null ,
$advancedOptions = null
) {
$alterList = [];
2015-12-18 07:18:01 +13:00
if ( $newFields ) {
foreach ( $newFields as $fieldName => $fieldSpec ) {
2016-01-21 09:49:28 +13:00
$alterList [] = " ADD \" $fieldName\ " $fieldSpec " ;
}
2015-12-18 07:18:01 +13:00
}
if ( $alteredFields ) {
foreach ( $alteredFields as $indexName => $indexSpec ) {
2016-01-21 09:49:28 +13:00
$val = $this -> alterTableAlterColumn ( $table , $indexName , $indexSpec );
2015-12-18 07:18:01 +13:00
if ( ! empty ( $val )) {
$alterList [] = $val ;
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
}
}
2016-01-21 09:49:28 +13:00
//Do we need to do anything with the tablespaces?
if ( $alteredOptions && isset ( $advancedOptions [ 'tablespace' ])) {
2017-11-17 10:58:14 +13:00
$this -> createOrReplaceTablespace (
$advancedOptions [ 'tablespace' ][ 'name' ],
$advancedOptions [ 'tablespace' ][ 'location' ]
);
2016-01-21 09:49:28 +13:00
$this -> query ( " ALTER TABLE \" $table\ " SET TABLESPACE { $advancedOptions [ 'tablespace' ][ 'name' ]}; " );
}
//DB ABSTRACTION: we need to change the constraints to be a separate 'add' command,
//see http://www.postgresql.org/docs/8.1/static/sql-altertable.html
2017-11-17 10:58:14 +13:00
$alterIndexList = [];
2016-01-21 09:49:28 +13:00
//Pick up the altered indexes here:
$fieldList = $this -> fieldList ( $table );
2017-11-17 10:58:14 +13:00
$fulltexts = [];
$dropTriggers = [];
$triggers = [];
2015-12-18 07:18:01 +13:00
if ( $alteredIndexes ) {
2017-11-17 10:58:14 +13:00
foreach ( $alteredIndexes as $indexName => $indexSpec ) {
2016-01-21 09:49:28 +13:00
$indexNamePG = $this -> buildPostgresIndexName ( $table , $indexName );
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
if ( $indexSpec [ 'type' ] == 'fulltext' ) {
2016-01-21 09:49:28 +13:00
//For full text indexes, we need to drop the trigger, drop the index, AND drop the column
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
//Go and get the tsearch details:
$ts_details = $this -> fulltext ( $indexSpec , $table , $indexName );
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
//Drop this column if it already exists:
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
//No IF EXISTS option is available for Postgres <9.0
if ( array_key_exists ( $ts_details [ 'ts_name' ], $fieldList )) {
$fulltexts [] = " ALTER TABLE \" { $table } \" DROP COLUMN \" { $ts_details [ 'ts_name' ] } \" ; " ;
}
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
// We'll execute these later:
$triggerNamePG = $this -> buildPostgresTriggerName ( $table , $indexName );
$dropTriggers [] = " DROP TRIGGER IF EXISTS \" $triggerNamePG\ " ON \ " $table\ " ; " ;
$fulltexts [] = " ALTER TABLE \" { $table } \" ADD COLUMN { $ts_details [ 'fulltexts' ] } ; " ;
$triggers [] = $ts_details [ 'triggers' ];
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
// Create index action (including fulltext)
$alterIndexList [] = " DROP INDEX IF EXISTS \" $indexNamePG\ " ; " ;
2016-01-21 09:49:28 +13:00
$createIndex = $this -> getIndexSqlDefinition ( $table , $indexName , $indexSpec );
2017-11-17 10:58:14 +13:00
if ( $createIndex ) {
2015-12-18 07:18:01 +13:00
$alterIndexList [] = $createIndex ;
}
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
}
2016-01-21 09:49:28 +13:00
//Add the new indexes:
2015-12-18 07:18:01 +13:00
if ( $newIndexes ) {
foreach ( $newIndexes as $indexName => $indexSpec ) {
2016-01-21 09:49:28 +13:00
$indexNamePG = $this -> buildPostgresIndexName ( $table , $indexName );
2017-11-17 10:58:14 +13:00
//If we have a fulltext search request, then we need to create a special column
//for GiST searches
//Pick up the new indexes here:
if ( $indexSpec [ 'type' ] == 'fulltext' ) {
$ts_details = $this -> fulltext ( $indexSpec , $table , $indexName );
if ( ! isset ( $fieldList [ $ts_details [ 'ts_name' ]])) {
$fulltexts [] = " ALTER TABLE \" { $table } \" ADD COLUMN { $ts_details [ 'fulltexts' ] } ; " ;
$triggers [] = $ts_details [ 'triggers' ];
}
2016-01-21 09:49:28 +13:00
}
2017-11-17 10:58:14 +13:00
//Check that this index doesn't already exist:
$indexes = $this -> indexList ( $table );
2016-01-21 09:49:28 +13:00
if ( isset ( $indexes [ $indexName ])) {
$alterIndexList [] = " DROP INDEX IF EXISTS \" $indexNamePG\ " ; " ;
}
2017-11-17 10:58:14 +13:00
$createIndex = $this -> getIndexSqlDefinition ( $table , $indexName , $indexSpec );
if ( $createIndex ) {
2016-01-21 09:49:28 +13:00
$alterIndexList [] = $createIndex ;
}
2015-12-18 07:18:01 +13:00
}
}
2016-01-21 09:49:28 +13:00
if ( $alterList ) {
$alterations = implode ( " , \n " , $alterList );
$this -> query ( " ALTER TABLE \" $table\ " " . $alterations );
}
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
//Do we need to create a tablespace for this item?
if ( $advancedOptions && isset ( $advancedOptions [ 'extensions' ][ 'tablespace' ])) {
2017-11-17 10:58:14 +13:00
$extensions = $advancedOptions [ 'extensions' ];
2016-01-21 09:49:28 +13:00
$this -> createOrReplaceTablespace ( $extensions [ 'tablespace' ][ 'name' ], $extensions [ 'tablespace' ][ 'location' ]);
}
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
if ( $alteredOptions && isset ( $this -> class ) && isset ( $alteredOptions [ $this -> class ])) {
$this -> query ( sprintf ( " ALTER TABLE \" %s \" %s " , $table , $alteredOptions [ $this -> class ]));
2016-07-04 12:44:09 +12:00
DB :: alteration_message (
2016-01-21 09:49:28 +13:00
sprintf ( " Table %s options changed: %s " , $table , $alteredOptions [ $this -> class ]),
" changed "
);
}
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
//Create any fulltext columns and triggers here:
2017-11-17 10:58:14 +13:00
foreach ( $fulltexts as $fulltext ) {
$this -> query ( $fulltext );
2015-12-18 07:18:01 +13:00
}
2017-11-17 10:58:14 +13:00
foreach ( $dropTriggers as $dropTrigger ) {
$this -> query ( $dropTrigger );
2015-12-18 07:18:01 +13:00
}
2017-11-17 10:58:14 +13:00
foreach ( $triggers as $trigger ) {
$this -> query ( $trigger );
$triggerFields = $this -> triggerFieldsFromTrigger ( $trigger );
if ( $triggerFields ) {
//We need to run a simple query to force the database to update the triggered columns
$this -> query ( " UPDATE \" { $table } \" SET \" { $triggerFields [ 0 ] } \" = \" $triggerFields[0] \" ; " );
2016-01-21 09:49:28 +13:00
}
}
2015-12-18 07:18:01 +13:00
foreach ( $alterIndexList as $alteration ) {
$this -> query ( $alteration );
}
2016-01-21 09:49:28 +13:00
//If we have a partitioning requirement, we do that here:
if ( $advancedOptions && isset ( $advancedOptions [ 'partitions' ])) {
$this -> createOrReplacePartition ( $table , $advancedOptions [ 'partitions' ]);
}
//Lastly, clustering goes here:
if ( $advancedOptions && isset ( $advancedOptions [ 'cluster' ])) {
$clusterIndex = $this -> buildPostgresIndexName ( $table , $advancedOptions [ 'cluster' ]);
$this -> query ( " CLUSTER \" $table\ " USING \ " $clusterIndex\ " ; " );
} else {
//Check that clustering is not on this table, and if it is, remove it:
//This is really annoying. We need the oid of this table:
$stats = $this -> preparedQuery (
" SELECT relid FROM pg_stat_user_tables WHERE relname = ?; " ,
array ( $table )
) -> first ();
2017-11-17 10:58:14 +13:00
$oid = $stats [ 'relid' ];
2016-01-21 09:49:28 +13:00
//Now we can run a long query to get the clustered status:
//If anyone knows a better way to get the clustered status, then feel free to replace this!
2017-11-17 10:58:14 +13:00
$clustered = $this -> preparedQuery (
"
2016-01-21 09:28:37 +13:00
SELECT c2 . relname , i . indisclustered
FROM pg_catalog . pg_class c , pg_catalog . pg_class c2 , pg_catalog . pg_index i
WHERE c . oid = ? AND c . oid = i . indrelid AND i . indexrelid = c2 . oid AND indisclustered = 't' ; " ,
2016-01-21 09:49:28 +13:00
array ( $oid )
) -> first ();
if ( $clustered ) {
$this -> query ( " ALTER TABLE \" $table\ " SET WITHOUT CLUSTER ; " );
}
}
}
/*
* Creates an ALTER expression for a column in PostgreSQL
*
* @ param $tableName Name of the table to be altered
* @ param $colName Name of the column to be altered
* @ param $colSpec String which contains conditions for a column
* @ return string
*/
2015-12-18 07:18:01 +13:00
private function alterTableAlterColumn ( $tableName , $colName , $colSpec )
{
2016-01-21 09:49:28 +13:00
// First, we split the column specifications into parts
// TODO: this returns an empty array for the following string: int(11) not null auto_increment
2016-01-21 09:28:37 +13:00
// on second thoughts, why is an auto_increment field being passed through?
2019-09-26 10:25:41 +12:00
$pattern = '/^([\w(\,)]+)\s?((?:not\s)?null)?\s?(default\s[\w\.\'\\\\]+)?\s?(check\s[\w()\'",\s\\\\]+)?$/i' ;
2016-01-21 09:49:28 +13:00
preg_match ( $pattern , $colSpec , $matches );
2019-09-26 10:25:41 +12:00
// example value this regex is expected to parse:
// varchar(255) not null default 'SS\Test\Player' check ("ClassName" in ('SS\Test\Player', 'Player', null))
// split into:
// * varchar(255)
// * not null
// * default 'SS\Test\Player'
// * check ("ClassName" in ('SS\Test\Player', 'Player', null))
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
if ( sizeof ( $matches ) == 0 ) {
2015-12-18 07:18:01 +13:00
return '' ;
}
2017-11-17 10:58:14 +13:00
if ( $matches [ 1 ] == 'serial8' ) {
2015-12-18 07:18:01 +13:00
return '' ;
}
2016-01-21 09:49:28 +13:00
if ( isset ( $matches [ 1 ])) {
2017-07-26 08:04:48 +12:00
$alterCol = " ALTER COLUMN \" $colName\ " TYPE $matches [ 1 ] USING \ " $colName\ " :: $matches [ 1 ] \n " ;
2016-01-21 09:49:28 +13:00
// SET null / not null
if ( ! empty ( $matches [ 2 ])) {
$alterCol .= " , \n ALTER COLUMN \" $colName\ " SET $matches [ 2 ] " ;
}
// SET default (we drop it first, for reasons of precaution)
if ( ! empty ( $matches [ 3 ])) {
$alterCol .= " , \n ALTER COLUMN \" $colName\ " DROP DEFAULT " ;
$alterCol .= " , \n ALTER COLUMN \" $colName\ " SET $matches [ 3 ] " ;
}
// SET check constraint (The constraint HAS to be dropped)
2016-01-21 09:28:37 +13:00
$constraintName = " { $tableName } _ { $colName } _check " ;
$constraintExists = $this -> constraintExists ( $constraintName , false );
2016-01-21 09:49:28 +13:00
if ( isset ( $matches [ 4 ])) {
//Take this new constraint and see what's outstanding from the target table:
2017-11-17 10:58:14 +13:00
$constraint_bits = explode ( '(' , $matches [ 4 ]);
$constraint_values = trim ( $constraint_bits [ 2 ], ')' );
$constraint_values_bits = explode ( ',' , $constraint_values );
$default = trim ( $constraint_values_bits [ 0 ], " ' " );
2016-01-21 09:49:28 +13:00
//Now go and convert anything that's not in this list to 'Page'
//We have to run this as a query, not as part of the alteration queries due to the way they are constructed.
2017-11-17 10:58:14 +13:00
$updateConstraint = '' ;
$updateConstraint .= " UPDATE \" { $tableName } \" SET \" $colName\ " = '$default' WHERE \ " $colName\ " NOT IN ( $constraint_values ); " ;
2016-01-21 09:49:28 +13:00
if ( $this -> hasTable ( " { $tableName } _Live " )) {
2017-11-17 10:58:14 +13:00
$updateConstraint .= " UPDATE \" { $tableName } _Live \" SET \" $colName\ " = '$default' WHERE \ " $colName\ " NOT IN ( $constraint_values ); " ;
2016-01-21 09:49:28 +13:00
}
2019-09-26 10:25:41 +12:00
if ( $this -> hasTable ( " { $tableName } _Versions " )) {
$updateConstraint .= " UPDATE \" { $tableName } _Versions \" SET \" $colName\ " = '$default' WHERE \ " $colName\ " NOT IN ( $constraint_values ); " ;
2016-01-21 09:49:28 +13:00
}
$this -> query ( $updateConstraint );
}
//First, delete any existing constraint on this column, even if it's no longer an enum
2016-01-21 09:28:37 +13:00
if ( $constraintExists ) {
$alterCol .= " , \n DROP CONSTRAINT \" { $constraintName } \" " ;
2016-01-21 09:49:28 +13:00
}
//Now create the constraint (if we've asked for one)
if ( ! empty ( $matches [ 4 ])) {
2016-01-21 09:28:37 +13:00
$alterCol .= " , \n ADD CONSTRAINT \" { $constraintName } \" $matches[4] " ;
2016-01-21 09:49:28 +13:00
}
}
return isset ( $alterCol ) ? $alterCol : '' ;
}
2015-12-18 07:18:01 +13:00
public function renameTable ( $oldTableName , $newTableName )
{
2019-09-26 10:25:41 +12:00
$constraints = $this -> getConstraintForTable ( $oldTableName );
2016-01-21 09:49:28 +13:00
$this -> query ( " ALTER TABLE \" $oldTableName\ " RENAME TO \ " $newTableName\ " " );
2019-09-26 10:25:41 +12:00
if ( $constraints ) {
foreach ( $constraints as $old ) {
$new = preg_replace ( '/^' . $oldTableName . '/' , $newTableName , $old );
$this -> query ( " ALTER TABLE \" $newTableName\ " RENAME CONSTRAINT \ " $old\ " TO \ " $new\ " ; " );
}
}
2016-01-21 09:49:28 +13:00
unset ( self :: $cached_fieldlists [ $oldTableName ]);
2019-09-26 10:25:41 +12:00
unset ( self :: $cached_constraints [ $oldTableName ]);
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
public function checkAndRepairTable ( $tableName )
{
2016-01-21 09:49:28 +13:00
$this -> query ( " VACUUM FULL ANALYZE \" $tableName\ " " );
$this -> query ( " REINDEX TABLE \" $tableName\ " " );
return true ;
}
2015-12-18 07:18:01 +13:00
public function createField ( $table , $field , $spec )
{
2016-01-21 09:49:28 +13:00
$this -> query ( " ALTER TABLE \" $table\ " ADD \ " $field\ " $spec " );
}
/**
* Change the database type of the given field .
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName The name of the tbale the field is in .
* @ param string $fieldName The name of the field to change .
* @ param string $fieldSpec The new field specification
*/
2015-12-18 07:18:01 +13:00
public function alterField ( $tableName , $fieldName , $fieldSpec )
{
2016-01-21 09:49:28 +13:00
$this -> query ( " ALTER TABLE \" $tableName\ " CHANGE \ " $fieldName\ " \ " $fieldName\ " $fieldSpec " );
}
2015-12-18 07:18:01 +13:00
public function renameField ( $tableName , $oldName , $newName )
{
2016-01-21 09:49:28 +13:00
$fieldList = $this -> fieldList ( $tableName );
if ( array_key_exists ( $oldName , $fieldList )) {
$this -> query ( " ALTER TABLE \" $tableName\ " RENAME COLUMN \ " $oldName\ " TO \ " $newName\ " " );
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
//Remove this from the cached list:
unset ( self :: $cached_fieldlists [ $tableName ]);
}
}
2015-12-18 07:18:01 +13:00
public function fieldList ( $table )
{
2016-01-21 09:49:28 +13:00
//Query from http://www.alberton.info/postgresql_meta_info.html
//This gets us more information than we need, but I've included it all for the moment....
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
//if(!isset(self::$cached_fieldlists[$table])){
2017-11-17 10:58:14 +13:00
$fields = $this -> preparedQuery (
"
2016-01-21 09:28:37 +13:00
SELECT ordinal_position , column_name , data_type , column_default ,
is_nullable , character_maximum_length , numeric_precision , numeric_scale
FROM information_schema . columns WHERE table_name = ? and table_schema = ?
ORDER BY ordinal_position ; " ,
2017-11-17 10:58:14 +13:00
array ( $table , $this -> database -> currentSchema ())
);
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
$output = array ();
2015-12-18 07:18:01 +13:00
if ( $fields ) {
foreach ( $fields as $field ) {
2016-01-21 09:49:28 +13:00
switch ( $field [ 'data_type' ]) {
case 'character varying' :
//Check to see if there's a constraint attached to this column:
//$constraint=$this->query("SELECT conname,pg_catalog.pg_get_constraintdef(r.oid, true) FROM pg_catalog.pg_constraint r WHERE r.contype = 'c' AND conname='" . $table . '_' . $field['column_name'] . "_check' ORDER BY 1;")->first();
$constraint = $this -> constraintExists ( $table . '_' . $field [ 'column_name' ] . '_check' );
if ( $constraint ) {
//Now we need to break this constraint text into bits so we can see what we have:
//Examples:
//CHECK ("CanEditType"::text = ANY (ARRAY['LoggedInUsers'::character varying, 'OnlyTheseUsers'::character varying, 'Inherit'::character varying]::text[]))
//CHECK ("ClassName"::text = 'PageComment'::text)
//TODO: replace all this with a regular expression!
2017-11-17 10:58:14 +13:00
$value = $constraint [ 'pg_get_constraintdef' ];
$value = substr ( $value , strpos ( $value , '=' ));
$value = str_replace ( " '' " , " ' " , $value );
$in_value = false ;
$constraints = array ();
$current_value = '' ;
for ( $i = 0 ; $i < strlen ( $value ); $i ++ ) {
$char = substr ( $value , $i , 1 );
2015-12-18 07:18:01 +13:00
if ( $in_value ) {
2017-11-17 10:58:14 +13:00
$current_value .= $char ;
2015-12-18 07:18:01 +13:00
}
2017-11-17 10:58:14 +13:00
if ( $char == " ' " ) {
2015-12-18 07:18:01 +13:00
if ( ! $in_value ) {
2017-11-17 10:58:14 +13:00
$in_value = true ;
2015-12-18 07:18:01 +13:00
} else {
2017-11-17 10:58:14 +13:00
$in_value = false ;
$constraints [] = substr ( $current_value , 0 , - 1 );
$current_value = '' ;
2016-01-21 09:49:28 +13:00
}
}
}
2017-11-17 10:58:14 +13:00
if ( sizeof ( $constraints ) > 0 ) {
2016-01-21 09:49:28 +13:00
//Get the default:
2017-11-17 10:58:14 +13:00
$default = trim ( substr (
$field [ 'column_default' ],
0 ,
strpos ( $field [ 'column_default' ], '::' )
), " ' " );
$output [ $field [ 'column_name' ]] = $this -> enum ( array (
'default' => $default ,
'name' => $field [ 'column_name' ],
'enums' => $constraints
));
2016-01-21 09:49:28 +13:00
}
} else {
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'varchar(' . $field [ 'character_maximum_length' ] . ')' ;
2016-01-21 09:49:28 +13:00
}
break ;
case 'numeric' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'decimal(' . $field [ 'numeric_precision' ] . ',' . $field [ 'numeric_scale' ] . ') default ' . floatval ( $field [ 'column_default' ]);
2016-01-21 09:49:28 +13:00
break ;
case 'integer' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'integer default ' . ( int ) $field [ 'column_default' ];
2016-01-21 09:49:28 +13:00
break ;
case 'timestamp without time zone' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'timestamp' ;
2016-01-21 09:49:28 +13:00
break ;
case 'smallint' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'smallint default ' . ( int ) $field [ 'column_default' ];
2016-01-21 09:49:28 +13:00
break ;
case 'time without time zone' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'time' ;
2016-01-21 09:49:28 +13:00
break ;
case 'double precision' :
2017-11-17 10:58:14 +13:00
$output [ $field [ 'column_name' ]] = 'float' ;
2016-01-21 09:49:28 +13:00
break ;
default :
$output [ $field [ 'column_name' ]] = $field ;
}
2015-12-18 07:18:01 +13:00
}
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
2016-01-21 09:28:37 +13:00
// self::$cached_fieldlists[$table]=$output;
2016-01-21 09:49:28 +13:00
//}
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
//return self::$cached_fieldlists[$table];
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
return $output ;
}
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
public function clearCachedFieldlist ( $tableName = false )
2015-12-18 07:18:01 +13:00
{
if ( $tableName ) {
unset ( self :: $cached_fieldlists [ $tableName ]);
} else {
2017-11-17 10:58:14 +13:00
self :: $cached_fieldlists = array ();
2015-12-18 07:18:01 +13:00
}
2016-01-21 09:49:28 +13:00
return true ;
}
/**
* Create an index on a table .
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName The name of the table .
* @ param string $indexName The name of the index .
* @ param string $indexSpec The specification of the index , see Database :: requireIndex () for more details .
*/
2015-12-18 07:18:01 +13:00
public function createIndex ( $tableName , $indexName , $indexSpec )
{
2016-01-21 09:49:28 +13:00
$createIndex = $this -> getIndexSqlDefinition ( $tableName , $indexName , $indexSpec );
2015-12-18 07:18:01 +13:00
if ( $createIndex !== false ) {
$this -> query ( $createIndex );
}
2016-01-21 09:49:28 +13:00
}
2016-07-04 13:49:59 +12:00
protected function getIndexSqlDefinition ( $tableName , $indexName , $indexSpec )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
//TODO: create table partition support
//TODO: create clustering options
//NOTE: it is possible for *_renamed tables to have indexes whose names are not updates
//Therefore, we now check for the existance of indexes before we create them.
//This is techically a bug, since new tables will not be indexed.
// Determine index name
$tableCol = $this -> buildPostgresIndexName ( $tableName , $indexName );
//Misc options first:
$fillfactor = $where = '' ;
if ( isset ( $indexSpec [ 'fillfactor' ])) {
$fillfactor = 'WITH (FILLFACTOR = ' . $indexSpec [ 'fillfactor' ] . ')' ;
}
if ( isset ( $indexSpec [ 'where' ])) {
$where = 'WHERE ' . $indexSpec [ 'where' ];
}
//create a type-specific index
// NOTE: hash should be removed. This is only here to demonstrate how other indexes can be made
2016-07-04 12:44:09 +12:00
// NOTE: Quote the index name to preserve case sensitivity
2016-01-21 09:49:28 +13:00
switch ( $indexSpec [ 'type' ]) {
case 'fulltext' :
// @see fulltext() for the definition of the trigger that ts_$IndexName uses for fulltext searching
$clusterMethod = PostgreSQLDatabase :: default_fts_cluster_method ();
$spec = " create index \" $tableCol\ " ON \ " $tableName\ " USING $clusterMethod ( \ " ts_ " . $indexName . " \" ) $fillfactor $where " ;
break ;
case 'unique' :
2017-05-19 14:49:50 +01:00
$spec = " create unique index \" $tableCol\ " ON \ " $tableName\ " ( " . $this->implodeColumnList ( $indexSpec['columns'] ) . " ) $fillfactor $where " ;
2016-01-21 09:49:28 +13:00
break ;
case 'btree' :
2017-05-19 14:49:50 +01:00
$spec = " create index \" $tableCol\ " ON \ " $tableName\ " USING btree ( " . $this->implodeColumnList ( $indexSpec['columns'] ) . " ) $fillfactor $where " ;
2016-01-21 09:49:28 +13:00
break ;
case 'hash' :
//NOTE: this is not a recommended index type
2017-05-19 14:49:50 +01:00
$spec = " create index \" $tableCol\ " ON \ " $tableName\ " USING hash ( " . $this->implodeColumnList ( $indexSpec['columns'] ) . " ) $fillfactor $where " ;
2016-01-21 09:49:28 +13:00
break ;
case 'index' :
2017-11-17 10:58:14 +13:00
//'index' is the same as default, just a normal index with the default type decided by the database.
2016-01-21 09:49:28 +13:00
default :
2017-05-19 14:49:50 +01:00
$spec = " create index \" $tableCol\ " ON \ " $tableName\ " ( " . $this->implodeColumnList ( $indexSpec['columns'] ) . " ) $fillfactor $where " ;
2016-01-21 09:49:28 +13:00
}
return trim ( $spec ) . ';' ;
}
2015-12-18 07:18:01 +13:00
public function alterIndex ( $tableName , $indexName , $indexSpec )
{
2016-01-21 09:49:28 +13:00
$indexSpec = trim ( $indexSpec );
if ( $indexSpec [ 0 ] != '(' ) {
list ( $indexType , $indexFields ) = explode ( ' ' , $indexSpec , 2 );
} else {
2016-07-04 12:44:09 +12:00
$indexType = null ;
2016-01-21 09:49:28 +13:00
$indexFields = $indexSpec ;
}
if ( ! $indexType ) {
$indexType = " index " ;
}
$this -> query ( " DROP INDEX \" $indexName\ " " );
$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
2017-11-17 10:58:14 +13:00
* @ param string $table
2016-01-21 09:49:28 +13:00
* @ return array List of columns
*/
2017-11-17 10:58:14 +13:00
protected function extractTriggerColumns ( $triggerName , $table )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
$trigger = $this -> preparedQuery (
2017-11-17 10:58:14 +13:00
" SELECT t.tgargs
FROM pg_catalog . pg_trigger t
INNER JOIN pg_catalog . pg_class c ON c . oid = t . tgrelid
INNER JOIN pg_catalog . pg_namespace n ON n . oid = c . relnamespace
WHERE c . relname = ?
AND n . nspname = ?
AND t . tgname = ? " ,
[
$table ,
$this -> database -> currentSchema (),
$triggerName
]
2016-01-21 09:49:28 +13:00
) -> first ();
2017-11-17 10:58:14 +13:00
// Convert stream to string
if ( is_resource ( $trigger [ 'tgargs' ])) {
$trigger [ 'tgargs' ] = stream_get_contents ( $trigger [ 'tgargs' ]);
}
2016-01-21 09:49:28 +13:00
2017-11-17 10:58:14 +13:00
if ( strpos ( $trigger [ 'tgargs' ], " \000 " ) !== false ) {
// Option 1: output as a string (PDO)
$argList = array_filter ( explode ( " \000 " , $trigger [ 'tgargs' ]));
2016-01-21 09:49:28 +13:00
} else {
2017-11-17 10:58:14 +13:00
// Option 2: hex-encoded (pg_sql non-pdo)
2016-01-21 09:49:28 +13:00
$bytes = str_split ( $trigger [ 'tgargs' ], 2 );
$argList = array ();
$nextArg = " " ;
foreach ( $bytes as $byte ) {
2022-04-21 14:14:28 +12:00
if ( $byte == '\x' ) {
continue ;
} elseif ( $byte == " 00 " ) {
2016-01-21 09:49:28 +13:00
$argList [] = $nextArg ;
$nextArg = " " ;
} else {
$nextArg .= chr ( hexdec ( $byte ));
}
}
}
// Drop first two arguments (trigger name and config name) and implode into nice list
return array_slice ( $argList , 2 );
}
2015-12-18 07:18:01 +13:00
public function indexList ( $table )
{
2016-01-21 09:49:28 +13:00
//Retrieve a list of indexes for the specified table
2017-11-17 10:58:14 +13:00
$indexes = $this -> preparedQuery (
"
2016-01-21 09:28:37 +13:00
SELECT tablename , indexname , indexdef
FROM pg_catalog . pg_indexes
WHERE tablename = ? AND schemaname = ? ; " ,
2016-01-21 09:49:28 +13:00
array ( $table , $this -> database -> currentSchema ())
);
$indexList = array ();
foreach ( $indexes as $index ) {
// Key for the indexList array. Differs from other DB implementations, which is why
// requireIndex() needed to be overridden
$indexName = $index [ 'indexname' ];
//We don't actually need the entire created command, just a few bits:
$type = '' ;
//Check for uniques:
2017-11-17 10:58:14 +13:00
if ( substr ( $index [ 'indexdef' ], 0 , 13 ) == 'CREATE UNIQUE' ) {
2016-01-21 09:49:28 +13:00
$type = 'unique' ;
}
//check for hashes, btrees etc:
2017-11-17 10:58:14 +13:00
if ( strpos ( strtolower ( $index [ 'indexdef' ]), 'using hash ' ) !== false ) {
2016-01-21 09:49:28 +13:00
$type = 'hash' ;
}
//TODO: Fix me: btree is the default index type:
//if(strpos(strtolower($index['indexdef']), 'using btree ')!==false)
2016-01-21 09:28:37 +13:00
// $prefix='using btree ';
2016-01-21 09:49:28 +13:00
2017-11-17 10:58:14 +13:00
if ( strpos ( strtolower ( $index [ 'indexdef' ]), 'using rtree ' ) !== false ) {
2016-01-21 09:49:28 +13:00
$type = 'rtree' ;
}
// For fulltext indexes we need to extract the columns from another source
if ( stristr ( $index [ 'indexdef' ], 'using gin' )) {
$type = 'fulltext' ;
// Extract trigger information from postgres
$triggerName = preg_replace ( '/^ix_/' , 'ts_' , $index [ 'indexname' ]);
2017-11-17 10:58:14 +13:00
$columns = $this -> extractTriggerColumns ( $triggerName , $table );
2016-01-21 09:49:28 +13:00
$columnString = $this -> implodeColumnList ( $columns );
} else {
$columnString = $this -> quoteColumnSpecString ( $index [ 'indexdef' ]);
}
2017-05-19 14:49:50 +01:00
$indexList [ $indexName ] = array (
2016-01-21 09:49:28 +13:00
'name' => $indexName , // Not the correct name in the PHP, as this will be a mangled postgres-unique code
2017-05-19 14:49:50 +01:00
'columns' => $this -> explodeColumnString ( $columnString ),
'type' => $type ? : 'index' ,
);
2016-01-21 09:49:28 +13:00
}
return $indexList ;
}
2015-12-18 07:18:01 +13:00
public function tableList ()
{
2016-01-21 09:49:28 +13:00
$tables = array ();
$result = $this -> preparedQuery (
" SELECT tablename FROM pg_catalog.pg_tables WHERE schemaname = ? AND tablename NOT ILIKE 'pg \\ \ _%' AND tablename NOT ILIKE 'sql \\ \ _%' " ,
array ( $this -> database -> currentSchema ())
);
foreach ( $result as $record ) {
$table = reset ( $record );
$tables [ strtolower ( $table )] = $table ;
}
return $tables ;
}
/**
* Find out what the constraint information is , given a constraint name .
* We also cache this result , so the next time we don ' t need to do a
* query all over again .
*
* @ param string $constraint
2016-01-21 09:28:37 +13:00
* @ param bool $cache Flag whether a cached version should be used . Set to false to cache bust .
* @ return false | array Either false , if the constraint doesn ' t exist , or an array
* with the keys conname and pg_get_constraintdef
2016-01-21 09:49:28 +13:00
*/
2016-01-21 09:28:37 +13:00
protected function constraintExists ( $constraint , $cache = true )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:28:37 +13:00
if ( ! $cache || ! isset ( self :: $cached_constraints [ $constraint ])) {
2017-11-17 10:58:14 +13:00
$value = $this -> preparedQuery (
"
2016-01-21 09:28:37 +13:00
SELECT conname , pg_catalog . pg_get_constraintdef ( r . oid , true )
FROM pg_catalog . pg_constraint r
INNER JOIN pg_catalog . pg_namespace n
ON r . connamespace = n . oid
WHERE r . contype = 'c' AND conname = ? AND n . nspname = ?
ORDER BY 1 ; " ,
array ( $constraint , $this -> database -> currentSchema ())
2016-01-21 09:49:28 +13:00
) -> first ();
2016-01-21 09:28:37 +13:00
if ( ! $cache ) {
return $value ;
}
self :: $cached_constraints [ $constraint ] = $value ;
2016-01-21 09:49:28 +13:00
}
return self :: $cached_constraints [ $constraint ];
}
2019-09-26 10:25:41 +12:00
/**
* Retrieve a list of constraints for the provided table name .
* @ param string $tableName
* @ return array
*/
private function getConstraintForTable ( $tableName )
{
// Note the PostgreSQL `like` operator is case sensitive
$constraints = $this -> preparedQuery (
"
SELECT conname
FROM pg_catalog . pg_constraint r
INNER JOIN pg_catalog . pg_namespace n
ON r . connamespace = n . oid
WHERE r . contype = 'c' AND conname like ? AND n . nspname = ?
ORDER BY 1 ; " ,
array ( $tableName . '_%' , $this -> database -> currentSchema ())
) -> column ( 'conname' );
return $constraints ;
}
2016-01-21 09:49:28 +13:00
/**
* A function to return the field names and datatypes for the particular table
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName
* @ return array List of columns an an associative array with the keys Column and DataType
*/
2015-12-18 07:18:01 +13:00
public function tableDetails ( $tableName )
{
2016-01-21 09:49:28 +13:00
$query = " SELECT a.attname as \" Column \" , pg_catalog.format_type(a.atttypid, a.atttypmod) as \" Datatype \"
2016-01-21 09:28:37 +13:00
FROM pg_catalog . pg_attribute a
WHERE a . attnum > 0 AND NOT a . attisdropped AND a . attrelid = (
SELECT c . oid
FROM pg_catalog . pg_class c
LEFT JOIN pg_catalog . pg_namespace n ON n . oid = c . relnamespace
WHERE c . relname = ? AND pg_catalog . pg_table_is_visible ( c . oid ) AND n . nspname = ?
); " ;
2013-04-03 17:27:11 +13:00
2016-01-21 09:28:37 +13:00
$result = $this -> preparedQuery (
$query ,
array ( $tableName , $this -> database -> currentSchema ())
);
2016-01-21 09:49:28 +13:00
$table = array ();
2016-01-21 09:28:37 +13:00
foreach ( $result as $row ) {
2016-01-21 09:49:28 +13:00
$table [] = array (
'Column' => $row [ 'Column' ],
'DataType' => $row [ 'DataType' ]
);
}
return $table ;
}
/**
* Pass a legit trigger name and it will be dropped
* This assumes that the trigger has been named in a unique fashion
*
* @ param string $triggerName Name of the trigger
* @ param string $tableName Name of the table
*/
2015-12-18 07:18:01 +13:00
protected function dropTrigger ( $triggerName , $tableName )
{
2017-11-17 10:58:14 +13:00
$exists = $this -> preparedQuery (
"
2016-01-21 09:28:37 +13:00
SELECT trigger_name
FROM information_schema . triggers
WHERE trigger_name = ? AND trigger_schema = ? ; " ,
2016-01-21 09:49:28 +13:00
array ( $triggerName , $this -> database -> currentSchema ())
) -> first ();
if ( $exists ) {
$this -> query ( " DROP trigger IF EXISTS $triggerName ON \" $tableName\ " ; " );
}
}
/**
* This will return the fields that the trigger is monitoring
*
* @ param string $trigger Name of the trigger
* @ return array
*/
2015-12-18 07:18:01 +13:00
protected function triggerFieldsFromTrigger ( $trigger )
{
2016-01-21 09:49:28 +13:00
if ( $trigger ) {
2017-11-17 10:58:14 +13:00
$tsvector = 'tsvector_update_trigger' ;
$ts_pos = strpos ( $trigger , $tsvector );
$details = trim ( substr ( $trigger , $ts_pos + strlen ( $tsvector )), '();' );
2016-01-21 09:49:28 +13:00
//Now split this into bits:
2017-11-17 10:58:14 +13:00
$bits = explode ( ',' , $details );
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
$fields = $bits [ 2 ];
2015-12-18 07:18:01 +13:00
2017-11-17 10:58:14 +13:00
$field_bits = explode ( ',' , str_replace ( '"' , '' , $fields ));
$result = array ();
2015-12-18 07:18:01 +13:00
foreach ( $field_bits as $field_bit ) {
2017-11-17 10:58:14 +13:00
$result [] = trim ( $field_bit );
2015-12-18 07:18:01 +13:00
}
2016-01-21 09:49:28 +13:00
return $result ;
} else {
return false ;
}
}
/**
* Return a boolean type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function boolean ( $values )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
$default = $values [ 'default' ] ? '1' : '0' ;
2016-07-04 12:44:09 +12:00
return " smallint default { $default } " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a date type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2015-12-18 07:18:01 +13:00
public function date ( $values )
{
2016-07-04 12:44:09 +12:00
return " date " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a decimal type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function decimal ( $values )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
// Avoid empty strings being put in the db
if ( $values [ 'precision' ] == '' ) {
$precision = 1 ;
} else {
$precision = $values [ 'precision' ];
}
$defaultValue = '' ;
if ( isset ( $values [ 'default' ]) && is_numeric ( $values [ 'default' ])) {
$defaultValue = ' default ' . floatval ( $values [ 'default' ]);
}
2016-07-04 12:44:09 +12:00
return " decimal( $precision ) $defaultValue " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a enum type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2015-12-18 07:18:01 +13:00
public function enum ( $values )
{
2016-07-04 12:44:09 +12:00
$default = " default ' { $values [ 'default' ] } ' " ;
2017-11-17 10:58:14 +13:00
return " varchar(255) " . $default . " check ( \" " . $values [ 'name' ] . " \" in (' " . implode (
'\', \'' ,
$values [ 'enums' ]
) . " ', null)) " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a float type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function float ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return " float " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a float type - formatted string cause double is not supported
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function double ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return $this -> float ( $values );
2016-01-21 09:49:28 +13:00
}
/**
* Return a int type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function int ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return " integer default " . ( int ) $values [ 'default' ];
2016-01-21 09:49:28 +13:00
}
/**
* Return a bigint type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function bigint ( $values )
2015-12-18 07:18:01 +13:00
{
2016-11-10 00:21:28 +00:00
return " bigint default " . ( int ) $values [ 'default' ];
2016-01-21 09:49:28 +13:00
}
/**
* Return a datetime type - formatted string
* For PostgreSQL , we simply return the word 'timestamp' , no other parameters are necessary
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function datetime ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return " timestamp " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a text type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function text ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return " text " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a time type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2015-12-18 07:18:01 +13:00
public function time ( $values )
{
2016-07-04 12:44:09 +12:00
return " time " ;
2016-01-21 09:49:28 +13:00
}
/**
* Return a varchar type - formatted string
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function varchar ( $values )
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
if ( ! isset ( $values [ 'precision' ])) {
$values [ 'precision' ] = 255 ;
}
2016-07-04 12:44:09 +12:00
return " varchar( { $values [ 'precision' ] } ) " ;
2016-01-21 09:49:28 +13:00
}
/*
* Return a 4 digit numeric type . MySQL has a proprietary 'Year' type .
* For Postgres , we ' ll use a 4 digit numeric
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2016-07-04 12:44:09 +12:00
public function year ( $values )
2015-12-18 07:18:01 +13:00
{
2016-07-04 12:44:09 +12:00
return " decimal(4,0) " ;
2016-01-21 09:49:28 +13:00
}
/**
* Create a fulltext search datatype for PostgreSQL
* This will also return a trigger to be applied to this table
*
* @ todo : create custom functions to allow weighted searches
*
* @ param array $this_index Index specification for the fulltext index
* @ param string $tableName
* @ param string $name
2016-07-04 12:44:09 +12:00
* @ return array
2016-01-21 09:49:28 +13:00
*/
2015-12-18 07:18:01 +13:00
protected function fulltext ( $this_index , $tableName , $name )
{
2016-01-21 09:49:28 +13:00
//For full text search, we need to create a column for the index
2017-05-19 14:49:50 +01:00
$columns = $this -> implodeColumnList ( $this_index [ 'columns' ]);
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
$fulltexts = " \" ts_ $name\ " tsvector " ;
$triggerName = $this -> buildPostgresTriggerName ( $tableName , $name );
$language = PostgreSQLDatabase :: search_language ();
2015-12-18 07:18:01 +13:00
2016-01-21 09:49:28 +13:00
$this -> dropTrigger ( $triggerName , $tableName );
$triggers = " CREATE TRIGGER \" $triggerName\ " BEFORE INSERT OR UPDATE
2016-01-21 09:28:37 +13:00
ON \ " $tableName\ " FOR EACH ROW EXECUTE PROCEDURE
tsvector_update_trigger ( \ " ts_ $name\ " , 'pg_catalog.$language' , $columns ); " ;
2013-04-03 17:27:11 +13:00
2016-01-21 09:49:28 +13:00
return array (
'name' => $name ,
'ts_name' => " ts_ { $name } " ,
'fulltexts' => $fulltexts ,
'triggers' => $triggers
);
}
2015-12-18 07:18:01 +13:00
public function IdColumn ( $asDbValue = false , $hasAutoIncPK = true )
{
if ( $asDbValue ) {
return 'bigint' ;
} else {
return 'serial8 not null' ;
}
2016-01-21 09:49:28 +13:00
}
2015-12-18 07:18:01 +13:00
public function hasTable ( $tableName )
{
2016-01-21 09:49:28 +13:00
$result = $this -> preparedQuery (
" SELECT tablename FROM pg_catalog.pg_tables WHERE schemaname = ? AND tablename = ?; " ,
array ( $this -> database -> currentSchema (), $tableName )
);
return ( $result -> numRecords () > 0 );
}
/**
* Returns the values of the given enum field
*
* @ todo Make a proper implementation
*
* @ param string $tableName Name of table to check
* @ param string $fieldName name of enum field to check
* @ return array List of enum values
*/
2015-12-18 07:18:01 +13:00
public function enumValuesForField ( $tableName , $fieldName )
{
2016-01-21 09:49:28 +13:00
//return array('SiteTree','Page');
$constraints = $this -> constraintExists ( " { $tableName } _ { $fieldName } _check " );
if ( $constraints ) {
return $this -> enumValuesFromConstraint ( $constraints [ 'pg_get_constraintdef' ]);
} else {
return array ();
}
}
/**
* Get the actual enum fields from the constraint value :
*
* @ param string $constraint
* @ return array
*/
2015-12-18 07:18:01 +13:00
protected function enumValuesFromConstraint ( $constraint )
{
2017-11-17 10:58:14 +13:00
$constraint = substr ( $constraint , strpos ( $constraint , 'ANY (ARRAY[' ) + 11 );
2016-01-21 09:49:28 +13:00
$constraint = substr ( $constraint , 0 , - 11 );
$constraints = array ();
$segments = explode ( ',' , $constraint );
foreach ( $segments as $this_segment ) {
$bits = preg_split ( '/ *:: */' , $this_segment );
array_unshift ( $constraints , trim ( $bits [ 0 ], " ' " ));
}
return $constraints ;
}
2015-12-18 07:18:01 +13:00
public function dbDataType ( $type )
{
2016-01-21 09:49:28 +13:00
$values = array (
'unsigned integer' => 'INT'
);
2015-12-18 07:18:01 +13:00
if ( isset ( $values [ $type ])) {
return $values [ $type ];
} else {
return '' ;
}
2016-01-21 09:49:28 +13:00
}
/*
* Given a tablespace and and location , either create a new one
* or update the existing one
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $name
* @ param string $location
*/
2015-12-18 07:18:01 +13:00
public function createOrReplaceTablespace ( $name , $location )
{
2016-01-21 09:49:28 +13:00
$existing = $this -> preparedQuery (
" SELECT spcname, spclocation FROM pg_tablespace WHERE spcname = ?; " ,
array ( $name )
) -> first ();
//NOTE: this location must be empty for this to work
//We can't seem to change the location of the tablespace through any ALTER commands :(
//If a tablespace with this name exists, but the location has changed, then drop the current one
//if($existing && $location!=$existing['spclocation'])
2016-01-21 09:28:37 +13:00
// DB::query("DROP TABLESPACE $name;");
2016-01-21 09:49:28 +13:00
//If this is a new tablespace, or we have dropped the current one:
if ( ! $existing || ( $existing && $location != $existing [ 'spclocation' ])) {
$this -> query ( " CREATE TABLESPACE $name LOCATION ' $location '; " );
}
}
/**
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $tableName
* @ param array $partitions
* @ param array $indexes
* @ param array $extensions
*/
2016-07-04 12:44:09 +12:00
public function createOrReplacePartition ( $tableName , $partitions , $indexes = [], $extensions = [])
2015-12-18 07:18:01 +13:00
{
2016-01-21 09:49:28 +13:00
//We need the plpgsql language to be installed for this to work:
$this -> createLanguage ( 'plpgsql' );
2017-11-17 10:58:14 +13:00
$trigger = 'CREATE OR REPLACE FUNCTION ' . $tableName . '_insert_trigger() RETURNS TRIGGER AS $$ BEGIN ' ;
$first = true ;
2016-01-21 09:49:28 +13:00
//Do we need to create a tablespace for this item?
if ( $extensions && isset ( $extensions [ 'tablespace' ])) {
$this -> createOrReplaceTablespace ( $extensions [ 'tablespace' ][ 'name' ], $extensions [ 'tablespace' ][ 'location' ]);
2017-11-17 10:58:14 +13:00
$tableSpace = ' TABLESPACE ' . $extensions [ 'tablespace' ][ 'name' ];
2016-01-21 09:49:28 +13:00
} else {
2017-11-17 10:58:14 +13:00
$tableSpace = '' ;
2016-01-21 09:49:28 +13:00
}
foreach ( $partitions as $partition_name => $partition_value ) {
//Check that this child table does not already exist:
if ( ! $this -> hasTable ( $partition_name )) {
2017-11-17 10:58:14 +13:00
$this -> query ( " CREATE TABLE \" $partition_name\ " ( CHECK ( " . str_replace(
'NEW.' ,
'' ,
$partition_value
) . " )) INHERITS ( \" $tableName\ " ) $tableSpace ; " );
2016-01-21 09:49:28 +13:00
} else {
//Drop the constraint, we will recreate in in the next line
2016-01-21 09:28:37 +13:00
$constraintName = " { $partition_name } _pkey " ;
$constraintExists = $this -> constraintExists ( $constraintName , false );
if ( $constraintExists ) {
$this -> query ( " ALTER TABLE \" $partition_name\ " DROP CONSTRAINT \ " { $constraintName } \" ; " );
2016-01-21 09:49:28 +13:00
}
$this -> dropTrigger ( strtolower ( 'trigger_' . $tableName . '_insert' ), $tableName );
}
$this -> query ( " ALTER TABLE \" $partition_name\ " ADD CONSTRAINT \ " { $partition_name } _pkey \" PRIMARY KEY ( \" ID \" ); " );
if ( $first ) {
2017-11-17 10:58:14 +13:00
$trigger .= 'IF' ;
$first = false ;
2016-01-21 09:49:28 +13:00
} else {
2017-11-17 10:58:14 +13:00
$trigger .= 'ELSIF' ;
2016-01-21 09:49:28 +13:00
}
$trigger .= " ( $partition_value ) THEN INSERT INTO \" $partition_name\ " VALUES ( NEW .* ); " ;
if ( $indexes ) {
// We need to propogate the indexes through to the child pages.
// Some of this code is duplicated, and could be tidied up
foreach ( $indexes as $name => $this_index ) {
2017-11-17 10:58:14 +13:00
if ( $this_index [ 'type' ] == 'fulltext' ) {
2016-01-21 09:49:28 +13:00
$fillfactor = $where = '' ;
if ( isset ( $this_index [ 'fillfactor' ])) {
$fillfactor = 'WITH (FILLFACTOR = ' . $this_index [ 'fillfactor' ] . ')' ;
}
if ( isset ( $this_index [ 'where' ])) {
$where = 'WHERE ' . $this_index [ 'where' ];
}
$clusterMethod = PostgreSQLDatabase :: default_fts_cluster_method ();
2017-11-17 10:58:14 +13:00
$this -> query ( " CREATE INDEX \" " . $this -> buildPostgresIndexName (
$partition_name ,
$this_index [ 'name' ]
) . " \" ON \" " . $partition_name . " \" USING $clusterMethod ( \" ts_ " . $name . " \" ) $fillfactor $where " );
2016-01-21 09:49:28 +13:00
$ts_details = $this -> fulltext ( $this_index , $partition_name , $name );
$this -> query ( $ts_details [ 'triggers' ]);
} else {
if ( is_array ( $this_index )) {
$index_name = $this_index [ 'name' ];
} else {
$index_name = trim ( $this_index , '()' );
}
$createIndex = $this -> getIndexSqlDefinition ( $partition_name , $index_name , $this_index );
if ( $createIndex !== false ) {
$this -> query ( $createIndex );
}
}
}
}
//Lastly, clustering goes here:
if ( $extensions && isset ( $extensions [ 'cluster' ])) {
$this -> query ( " CLUSTER \" $partition_name\ " USING \ " { $extensions [ 'cluster' ] } \" ; " );
}
}
$trigger .= 'ELSE RAISE EXCEPTION \'Value id out of range. Fix the ' . $tableName . '_insert_trigger() function!\'; END IF; RETURN NULL; END; $$ LANGUAGE plpgsql;' ;
$trigger .= 'CREATE TRIGGER trigger_' . $tableName . '_insert BEFORE INSERT ON "' . $tableName . '" FOR EACH ROW EXECUTE PROCEDURE ' . $tableName . '_insert_trigger();' ;
$this -> query ( $trigger );
}
/*
* This will create a language if it doesn ' t already exist .
* This is used by the createOrReplacePartition function , which needs plpgsql
2016-07-04 12:44:09 +12:00
*
2016-01-21 09:49:28 +13:00
* @ param string $language Language name
*/
2015-12-18 07:18:01 +13:00
public function createLanguage ( $language )
{
2016-01-21 09:49:28 +13:00
$result = $this -> preparedQuery (
" SELECT lanname FROM pg_language WHERE lanname = ?; " ,
array ( $language )
) -> first ();
if ( ! $result ) {
$this -> query ( " CREATE LANGUAGE $language ; " );
}
}
/**
* Return a set type - formatted string
* This is used for Multi - enum support , which isn ' t actually supported by Postgres .
* Throws a user error to show our lack of support , and return an " int " , specifically for sapphire
* tests that test multi - enums . This results in a test failure , but not crashing the test run .
*
* @ param array $values Contains a tokenised list of info about this data type
* @ return string
*/
2015-12-18 07:18:01 +13:00
public function set ( $values )
{
2016-01-21 09:49:28 +13:00
user_error ( " PostGreSQL does not support multi-enum " , E_USER_ERROR );
return " int " ;
}
2013-04-03 17:27:11 +13:00
}