2010-01-30 05:45:10 +00:00
< ? php
/**
* StatusNet , the distributed open - source microblogging tool
*
* Database schema utilities
*
* PHP version 5
*
* LICENCE : This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU Affero General Public License for more details .
*
* You should have received a copy of the GNU Affero General Public License
* along with this program . If not , see < http :// www . gnu . org / licenses />.
*
* @ category Database
* @ package StatusNet
* @ author Evan Prodromou < evan @ status . net >
* @ copyright 2009 StatusNet , Inc .
* @ license http :// www . fsf . org / licensing / licenses / agpl - 3.0 . html GNU Affero General Public License version 3.0
* @ link http :// status . net /
*/
if ( ! defined ( 'STATUSNET' )) {
exit ( 1 );
}
/**
* Class representing the database schema
*
* A class representing the database schema . Can be used to
* manipulate the schema -- especially for plugins and upgrade
* utilities .
*
* @ category Database
* @ package StatusNet
* @ author Evan Prodromou < evan @ status . net >
2010-05-15 03:56:40 +01:00
* @ author Brenda Wallace < shiny @ cpan . org >
2010-10-09 00:36:32 +01:00
* @ author Brion Vibber < brion @ status . net >
2010-01-30 05:45:10 +00:00
* @ license http :// www . fsf . org / licensing / licenses / agpl - 3.0 . html GNU Affero General Public License version 3.0
* @ link http :// status . net /
*/
class PgsqlSchema extends Schema
{
/**
2010-10-09 00:36:32 +01:00
* Returns a table definition array for the table
2010-01-30 05:45:10 +00:00
* in the schema with the given name .
*
* Throws an exception if the table is not found .
*
2010-10-09 00:36:32 +01:00
* @ param string $table Name of the table to get
2010-01-30 05:45:10 +00:00
*
2010-10-09 00:36:32 +01:00
* @ return array tabledef for that table .
2010-01-30 05:45:10 +00:00
*/
2010-10-09 00:36:32 +01:00
public function getTableDef ( $table )
2010-01-30 05:45:10 +00:00
{
2010-10-09 00:36:32 +01:00
$def = array ();
$hasKeys = false ;
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
// Pull column data from INFORMATION_SCHEMA
$columns = $this -> fetchMetaInfo ( $table , 'columns' , 'ordinal_position' );
if ( count ( $columns ) == 0 ) {
throw new SchemaTableMissingException ( " No such table: $table " );
2010-01-30 05:45:10 +00:00
}
2010-10-09 00:36:32 +01:00
foreach ( $columns as $row ) {
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
$name = $row [ 'column_name' ];
$field = array ();
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
// ??
list ( $type , $size ) = $this -> reverseMapType ( $row [ 'udt_name' ]);
$field [ 'type' ] = $type ;
if ( $size !== null ) {
$field [ 'size' ] = $size ;
}
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
if ( $type == 'char' || $type == 'varchar' ) {
if ( $row [ 'character_maximum_length' ] !== null ) {
$field [ 'length' ] = intval ( $row [ 'character_maximum_length' ]);
}
}
if ( $type == 'numeric' ) {
// Other int types may report these values, but they're irrelevant.
// Just ignore them!
if ( $row [ 'numeric_precision' ] !== null ) {
$field [ 'precision' ] = intval ( $row [ 'numeric_precision' ]);
}
if ( $row [ 'numeric_scale' ] !== null ) {
$field [ 'scale' ] = intval ( $row [ 'numeric_scale' ]);
}
}
if ( $row [ 'is_nullable' ] == 'NO' ) {
$field [ 'not null' ] = true ;
}
if ( $row [ 'column_default' ] !== null ) {
$field [ 'default' ] = $row [ 'column_default' ];
if ( $this -> isNumericType ( $type )) {
$field [ 'default' ] = intval ( $field [ 'default' ]);
}
}
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
$def [ 'fields' ][ $name ] = $field ;
}
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
// Pull constraint data from INFORMATION_SCHEMA
// @fixme also find multi-val indexes
// @fixme distinguish the primary key
// @fixme pull foreign key references
$keyColumns = $this -> fetchMetaInfo ( $table , 'key_column_usage' , 'constraint_name,ordinal_position' );
$keys = array ();
foreach ( $keyColumns as $row ) {
$keyName = $row [ 'constraint_name' ];
$keyCol = $row [ 'column_name' ];
if ( ! isset ( $keys [ $keyName ])) {
$keys [ $keyName ] = array ();
2010-01-30 05:45:10 +00:00
}
2010-10-09 00:36:32 +01:00
$keys [ $keyName ][] = $keyCol ;
}
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
foreach ( $keys as $keyName => $cols ) {
$def [ 'unique indexes' ][ $keyName ] = $cols ;
}
return $def ;
}
2010-01-30 05:45:10 +00:00
2010-10-09 00:36:32 +01:00
/**
* Pull some INFORMATION . SCHEMA data for the given table .
*
* @ param string $table
* @ return array of arrays
*/
function fetchMetaInfo ( $table , $infoTable , $orderBy = null )
{
$query = " SELECT * FROM information_schema.%s " .
" WHERE table_name='%s' " ;
$sql = sprintf ( $query , $infoTable , $table );
if ( $orderBy ) {
$sql .= ' ORDER BY ' . $orderBy ;
2010-01-30 05:45:10 +00:00
}
2010-10-09 00:36:32 +01:00
return $this -> fetchQueryData ( $sql );
2010-01-30 05:45:10 +00:00
}
/**
* Creates a table with the given names and columns .
*
* @ param string $name Name of the table
* @ param array $columns Array of ColumnDef objects
* for new table .
*
* @ return boolean success flag
*/
public function createTable ( $name , $columns )
{
$uniques = array ();
$primary = array ();
$indices = array ();
2010-05-17 02:50:37 +01:00
$onupdate = array ();
2010-01-30 05:45:10 +00:00
$sql = " CREATE TABLE $name ( \n " ;
for ( $i = 0 ; $i < count ( $columns ); $i ++ ) {
$cd =& $columns [ $i ];
if ( $i > 0 ) {
$sql .= " , \n " ;
}
$sql .= $this -> _columnSql ( $cd );
switch ( $cd -> key ) {
case 'UNI' :
$uniques [] = $cd -> name ;
break ;
case 'PRI' :
$primary [] = $cd -> name ;
break ;
case 'MUL' :
$indices [] = $cd -> name ;
break ;
}
}
if ( count ( $primary ) > 0 ) { // it really should be...
2010-05-15 04:31:54 +01:00
$sql .= " , \n PRIMARY KEY ( " . implode ( ',' , $primary ) . " ) " ;
2010-01-30 05:45:10 +00:00
}
$sql .= " ); " ;
2010-03-10 08:54:30 +00:00
foreach ( $uniques as $u ) {
$sql .= " \n CREATE index { $name } _ { $u } _idx ON { $name } ( $u ); " ;
}
2010-05-15 04:14:11 +01:00
foreach ( $indices as $i ) {
2010-05-15 04:31:54 +01:00
$sql .= " CREATE index { $name } _ { $i } _idx ON { $name } ( $i ) " ;
2010-05-15 04:14:11 +01:00
}
2010-01-30 05:45:10 +00:00
$res = $this -> conn -> query ( $sql );
if ( PEAR :: isError ( $res )) {
2010-05-15 03:56:40 +01:00
throw new Exception ( $res -> getMessage () . ' SQL was ' . $sql );
2010-01-30 05:45:10 +00:00
}
return true ;
}
2010-03-10 09:02:56 +00:00
/**
* Translate the ( mostly ) mysql - ish column types into somethings more standard
* @ param string column type
*
* @ return string postgres happy column type
*/
private function _columnTypeTranslation ( $type ) {
$map = array (
2010-05-15 03:56:40 +01:00
'datetime' => 'timestamp' ,
2010-03-10 09:02:56 +00:00
);
if ( ! empty ( $map [ $type ])) {
return $map [ $type ];
}
return $type ;
}
2010-01-30 05:45:10 +00:00
/**
* Modifies a column in the schema .
*
* The name must match an existing column and table .
*
* @ param string $table name of the table
* @ param ColumnDef $columndef new definition of the column .
*
* @ return boolean success flag
*/
public function modifyColumn ( $table , $columndef )
{
2010-05-17 02:50:37 +01:00
$sql = " ALTER TABLE $table ALTER COLUMN TYPE " .
2010-01-30 05:45:10 +00:00
$this -> _columnSql ( $columndef );
$res = $this -> conn -> query ( $sql );
if ( PEAR :: isError ( $res )) {
throw new Exception ( $res -> getMessage ());
}
return true ;
}
/**
* Ensures that a table exists with the given
* name and the given column definitions .
*
* If the table does not yet exist , it will
* create the table . If it does exist , it will
* alter the table to match the column definitions .
*
* @ param string $tableName name of the table
* @ param array $columns array of ColumnDef
* objects for the table
*
* @ return boolean success flag
*/
public function ensureTable ( $tableName , $columns )
{
// XXX: DB engine portability -> toilet
try {
$td = $this -> getTableDef ( $tableName );
2010-03-10 08:25:44 +00:00
2010-01-30 05:45:10 +00:00
} catch ( Exception $e ) {
if ( preg_match ( '/no such table/' , $e -> getMessage ())) {
return $this -> createTable ( $tableName , $columns );
} else {
throw $e ;
}
}
$cur = $this -> _names ( $td -> columns );
$new = $this -> _names ( $columns );
$toadd = array_diff ( $new , $cur );
$todrop = array_diff ( $cur , $new );
$same = array_intersect ( $new , $cur );
$tomod = array ();
foreach ( $same as $m ) {
$curCol = $this -> _byName ( $td -> columns , $m );
$newCol = $this -> _byName ( $columns , $m );
2010-05-15 03:56:40 +01:00
2010-01-30 05:45:10 +00:00
if ( ! $newCol -> equals ( $curCol )) {
2010-05-15 03:56:40 +01:00
// BIG GIANT TODO!
// stop it detecting different types and trying to modify on every page request
// $tomod[] = $newCol->name;
2010-01-30 05:45:10 +00:00
}
}
if ( count ( $toadd ) + count ( $todrop ) + count ( $tomod ) == 0 ) {
// nothing to do
return true ;
}
// For efficiency, we want this all in one
// query, instead of using our methods.
$phrase = array ();
foreach ( $toadd as $columnName ) {
$cd = $this -> _byName ( $columns , $columnName );
$phrase [] = 'ADD COLUMN ' . $this -> _columnSql ( $cd );
}
foreach ( $todrop as $columnName ) {
$phrase [] = 'DROP COLUMN ' . $columnName ;
}
foreach ( $tomod as $columnName ) {
$cd = $this -> _byName ( $columns , $columnName );
2010-05-17 02:50:37 +01:00
/* brute force */
$phrase [] = 'DROP COLUMN ' . $columnName ;
$phrase [] = 'ADD COLUMN ' . $this -> _columnSql ( $cd );
2010-01-30 05:45:10 +00:00
}
$sql = 'ALTER TABLE ' . $tableName . ' ' . implode ( ', ' , $phrase );
$res = $this -> conn -> query ( $sql );
if ( PEAR :: isError ( $res )) {
throw new Exception ( $res -> getMessage ());
}
return true ;
}
/**
2010-08-17 00:31:18 +01:00
* Return the proper SQL for creating or
* altering a column .
2010-01-30 05:45:10 +00:00
*
2010-08-17 00:31:18 +01:00
* Appropriate for use in CREATE TABLE or
* ALTER TABLE statements .
2010-01-30 05:45:10 +00:00
*
2010-08-17 00:31:18 +01:00
* @ param string $tableName
* @ param array $tableDef
* @ param string $columnName
* @ param array $cd column to create
2010-01-30 05:45:10 +00:00
*
2010-08-17 00:31:18 +01:00
* @ return string correct SQL for that column
2010-01-30 05:45:10 +00:00
*/
2010-08-17 00:31:18 +01:00
function columnSql ( $name , array $cd )
2010-01-30 05:45:10 +00:00
{
2010-08-17 00:31:18 +01:00
$line = array ();
$line [] = parent :: _columnSql ( $cd );
if ( $table [ 'foreign keys' ][ $name ]) {
foreach ( $table [ 'foreign keys' ][ $name ] as $foreignTable => $foreignColumn ) {
$line [] = 'references' ;
$line [] = $this -> quoteId ( $foreignTable );
$line [] = '(' . $this -> quoteId ( $foreignColumn ) . ')' ;
2010-01-30 05:45:10 +00:00
}
}
2010-08-17 00:31:18 +01:00
return implode ( ' ' , $line );
2010-01-30 05:45:10 +00:00
}
2010-08-17 00:31:18 +01:00
function mapType ( $column )
2010-01-30 05:45:10 +00:00
{
2010-08-17 00:31:18 +01:00
$map = array ( 'serial' => 'bigserial' , // FIXME: creates the wrong name for the sequence for some internal sequence-lookup function, so better fix this to do the real 'create sequence' dance.
'numeric' => 'decimal' ,
'datetime' => 'timestamp' ,
'blob' => 'bytea' );
$type = $column [ 'type' ];
if ( isset ( $map [ $type ])) {
$type = $map [ $type ];
2010-05-15 03:56:40 +01:00
}
2010-08-17 00:31:18 +01:00
if ( ! empty ( $column [ 'size' ])) {
$size = $column [ 'size' ];
if ( $type == 'integer' &&
in_array ( $size , array ( 'small' , 'big' ))) {
$type = $size . 'int' ;
}
2010-01-30 05:45:10 +00:00
}
2010-08-17 00:31:18 +01:00
return $type ;
}
// @fixme need name... :P
function typeAndSize ( $column )
{
if ( $column [ 'type' ] == 'enum' ) {
$vals = array_map ( array ( $this , 'quote' ), $column [ 'enum' ]);
return " text check ( $name in " . implode ( ',' , $vals ) . ')' ;
2010-01-30 05:45:10 +00:00
} else {
2010-08-17 00:31:18 +01:00
return parent :: typeAndSize ( $column );
2010-01-30 05:45:10 +00:00
}
}
2010-08-17 00:31:18 +01:00
2010-10-09 00:36:32 +01:00
/**
* Map a native type back to an independent type + size
*
* @ param string $type
* @ return array ( $type , $size ) -- $size may be null
*/
protected function reverseMapType ( $type )
{
$type = strtolower ( $type );
$map = array (
'int4' => array ( 'int' , null ),
'int8' => array ( 'int' , 'big' ),
'bytea' => array ( 'blob' , null ),
);
if ( isset ( $map [ $type ])) {
return $map [ $type ];
} else {
return array ( $type , null );
}
}
2010-01-30 05:45:10 +00:00
}