2008-08-01 18:46:14 +00:00
|
|
|
<?php
|
2009-05-03 18:07:57 +00:00
|
|
|
/*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* This software consists of voluntary contributions made by many individuals
|
|
|
|
* and is licensed under the LGPL. For more information, see
|
|
|
|
* <http://www.doctrine-project.org>.
|
|
|
|
*/
|
2008-08-01 18:46:14 +00:00
|
|
|
|
2009-01-22 19:38:10 +00:00
|
|
|
namespace Doctrine\DBAL\Platforms;
|
2008-08-01 18:46:14 +00:00
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
use Doctrine\DBAL\Schema\TableDiff;
|
|
|
|
|
2009-05-03 18:07:57 +00:00
|
|
|
/**
|
|
|
|
* PostgreSqlPlatform.
|
|
|
|
*
|
|
|
|
* @since 2.0
|
|
|
|
* @author Roman Borschel <roman@code-factory.org>
|
|
|
|
* @author Lukas Smith <smith@pooteeweet.org> (PEAR MDB2 library)
|
2009-11-03 08:58:03 +00:00
|
|
|
* @author Benjamin Eberlei <kontakt@beberlei.de>
|
2009-05-03 18:07:57 +00:00
|
|
|
*/
|
2009-01-22 19:38:10 +00:00
|
|
|
class PostgreSqlPlatform extends AbstractPlatform
|
2008-08-01 18:46:14 +00:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Returns part of a string.
|
|
|
|
*
|
|
|
|
* Note: Not SQL92, but common functionality.
|
|
|
|
*
|
|
|
|
* @param string $value the target $value the string or the string column.
|
|
|
|
* @param int $from extract from this characeter.
|
|
|
|
* @param int $len extract this amount of characters.
|
|
|
|
* @return string sql that extracts part of a string.
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getSubstringExpression($value, $from, $len = null)
|
|
|
|
{
|
|
|
|
if ($len === null) {
|
|
|
|
return 'SUBSTR(' . $value . ', ' . $from . ')';
|
|
|
|
} else {
|
|
|
|
return 'SUBSTR(' . $value . ', ' . $from . ', ' . $len . ')';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the SQL string to return the current system date and time.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getNowExpression()
|
|
|
|
{
|
|
|
|
return 'LOCALTIMESTAMP(0)';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* regexp
|
|
|
|
*
|
|
|
|
* @return string the regular expression operator
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getRegexpExpression()
|
|
|
|
{
|
|
|
|
return 'SIMILAR TO';
|
|
|
|
}
|
2010-02-11 22:38:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* returns the position of the first occurrence of substring $substr in string $str
|
|
|
|
*
|
|
|
|
* @param string $substr literal string to find
|
|
|
|
* @param string $str literal string
|
|
|
|
* @param int $pos position to start at, beginning of string by default
|
|
|
|
* @return integer
|
|
|
|
*/
|
|
|
|
public function getLocateExpression($str, $substr, $startPos = false)
|
|
|
|
{
|
|
|
|
if ($startPos !== false) {
|
|
|
|
$str = $this->getSubstringExpression($str, $startPos);
|
|
|
|
return 'CASE WHEN (POSITION('.$substr.' IN '.$str.') = 0) THEN 0 ELSE (POSITION('.$substr.' IN '.$str.') + '.($startPos-1).') END';
|
|
|
|
} else {
|
|
|
|
return 'POSITION('.$substr.' IN '.$str.')';
|
|
|
|
}
|
|
|
|
}
|
2008-08-01 18:46:14 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* parses a literal boolean value and returns
|
|
|
|
* proper sql equivalent
|
|
|
|
*
|
|
|
|
* @param string $value boolean value to be parsed
|
|
|
|
* @return string parsed boolean value
|
|
|
|
*/
|
2009-09-05 02:23:24 +00:00
|
|
|
/*public function parseBoolean($value)
|
2008-08-01 18:46:14 +00:00
|
|
|
{
|
|
|
|
return $value;
|
2009-09-05 02:23:24 +00:00
|
|
|
}*/
|
2008-08-02 17:41:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the platform supports sequences.
|
|
|
|
* Postgres has native support for sequences.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function supportsSequences()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-08-11 10:51:38 +00:00
|
|
|
/**
|
|
|
|
* Whether the platform supports database schemas.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function supportsSchemas()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-08-02 17:41:37 +00:00
|
|
|
/**
|
|
|
|
* Whether the platform supports identity columns.
|
|
|
|
* Postgres supports these through the SERIAL keyword.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function supportsIdentityColumns()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the platform prefers sequences for ID generation.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function prefersSequences()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2008-08-31 18:27:16 +00:00
|
|
|
public function getListDatabasesSql()
|
|
|
|
{
|
|
|
|
return 'SELECT datname FROM pg_database';
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2009-05-28 02:04:51 +00:00
|
|
|
public function getListSequencesSql($database)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
relname
|
|
|
|
FROM
|
|
|
|
pg_class
|
|
|
|
WHERE relkind = 'S' AND relnamespace IN
|
|
|
|
(SELECT oid FROM pg_namespace
|
|
|
|
WHERE nspname NOT LIKE 'pg_%' AND nspname != 'information_schema')";
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2008-08-31 18:27:16 +00:00
|
|
|
public function getListTablesSql()
|
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
c.relname AS table_name
|
|
|
|
FROM pg_class c, pg_user u
|
|
|
|
WHERE c.relowner = u.usesysid
|
|
|
|
AND c.relkind = 'r'
|
|
|
|
AND NOT EXISTS (SELECT 1 FROM pg_views WHERE viewname = c.relname)
|
|
|
|
AND c.relname !~ '^(pg_|sql_)'
|
|
|
|
UNION
|
|
|
|
SELECT c.relname AS table_name
|
|
|
|
FROM pg_class c
|
|
|
|
WHERE c.relkind = 'r'
|
|
|
|
AND NOT EXISTS (SELECT 1 FROM pg_views WHERE viewname = c.relname)
|
|
|
|
AND NOT EXISTS (SELECT 1 FROM pg_user WHERE usesysid = c.relowner)
|
|
|
|
AND c.relname !~ '^pg_'";
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2010-02-10 23:41:35 +00:00
|
|
|
public function getListViewsSql($database)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
2009-05-28 21:34:35 +00:00
|
|
|
return 'SELECT viewname, definition FROM pg_views';
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
|
|
|
public function getListTableForeignKeysSql($table, $database = null)
|
|
|
|
{
|
2009-12-03 13:17:56 +00:00
|
|
|
return "SELECT r.conname, pg_catalog.pg_get_constraintdef(r.oid, true) as condef
|
2009-05-28 21:34:35 +00:00
|
|
|
FROM pg_catalog.pg_constraint r
|
|
|
|
WHERE r.conrelid =
|
|
|
|
(
|
|
|
|
SELECT c.oid
|
|
|
|
FROM pg_catalog.pg_class c
|
|
|
|
LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
|
|
|
|
WHERE c.relname = '" . $table . "' AND pg_catalog.pg_table_is_visible(c.oid)
|
|
|
|
)
|
|
|
|
AND r.contype = 'f'";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getCreateViewSql($name, $sql)
|
|
|
|
{
|
|
|
|
return 'CREATE VIEW ' . $name . ' AS ' . $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getDropViewSql($name)
|
|
|
|
{
|
|
|
|
return 'DROP VIEW '. $name;
|
|
|
|
}
|
|
|
|
|
2009-05-27 18:54:40 +00:00
|
|
|
public function getListTableConstraintsSql($table)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
relname
|
|
|
|
FROM
|
|
|
|
pg_class
|
|
|
|
WHERE oid IN (
|
|
|
|
SELECT indexrelid
|
|
|
|
FROM pg_index, pg_class
|
2009-05-27 18:54:40 +00:00
|
|
|
WHERE pg_class.relname = '$table'
|
2008-08-31 18:27:16 +00:00
|
|
|
AND pg_class.oid = pg_index.indrelid
|
|
|
|
AND (indisunique = 't' OR indisprimary = 't')
|
|
|
|
)";
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2009-11-03 08:58:03 +00:00
|
|
|
/**
|
|
|
|
* @license New BSD License
|
|
|
|
* @link http://ezcomponents.org/docs/api/trunk/DatabaseSchema/ezcDbSchemaPgsqlReader.html
|
|
|
|
* @param string $table
|
|
|
|
* @return string
|
|
|
|
*/
|
2009-05-28 02:04:51 +00:00
|
|
|
public function getListTableIndexesSql($table)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
2009-11-03 08:58:03 +00:00
|
|
|
return "SELECT relname, pg_index.indisunique, pg_index.indisprimary,
|
|
|
|
pg_index.indkey, pg_index.indrelid
|
|
|
|
FROM pg_class, pg_index
|
|
|
|
WHERE oid IN (
|
2008-08-31 18:27:16 +00:00
|
|
|
SELECT indexrelid
|
|
|
|
FROM pg_index, pg_class
|
2009-11-03 08:58:03 +00:00
|
|
|
WHERE pg_class.relname='$table' AND pg_class.oid=pg_index.indrelid
|
|
|
|
) AND pg_index.indexrelid = oid";
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2009-05-27 18:54:40 +00:00
|
|
|
public function getListTableColumnsSql($table)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
a.attnum,
|
|
|
|
a.attname AS field,
|
|
|
|
t.typname AS type,
|
|
|
|
format_type(a.atttypid, a.atttypmod) AS complete_type,
|
|
|
|
a.attnotnull AS isnotnull,
|
|
|
|
(SELECT 't'
|
|
|
|
FROM pg_index
|
|
|
|
WHERE c.oid = pg_index.indrelid
|
|
|
|
AND pg_index.indkey[0] = a.attnum
|
|
|
|
AND pg_index.indisprimary = 't'
|
|
|
|
) AS pri,
|
|
|
|
(SELECT pg_attrdef.adsrc
|
|
|
|
FROM pg_attrdef
|
|
|
|
WHERE c.oid = pg_attrdef.adrelid
|
|
|
|
AND pg_attrdef.adnum=a.attnum
|
|
|
|
) AS default
|
|
|
|
FROM pg_attribute a, pg_class c, pg_type t
|
2009-05-27 18:54:40 +00:00
|
|
|
WHERE c.relname = '$table'
|
2008-08-31 18:27:16 +00:00
|
|
|
AND a.attnum > 0
|
|
|
|
AND a.attrelid = c.oid
|
|
|
|
AND a.atttypid = t.oid
|
|
|
|
ORDER BY a.attnum";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* create a new database
|
|
|
|
*
|
|
|
|
* @param string $name name of the database that should be created
|
|
|
|
* @throws PDOException
|
|
|
|
* @return void
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getCreateDatabaseSql($name)
|
|
|
|
{
|
2009-08-11 10:51:38 +00:00
|
|
|
return 'CREATE DATABASE ' . $name;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drop an existing database
|
|
|
|
*
|
|
|
|
* @param string $name name of the database that should be dropped
|
|
|
|
* @throws PDOException
|
|
|
|
* @access public
|
|
|
|
*/
|
|
|
|
public function getDropDatabaseSql($name)
|
|
|
|
{
|
2009-08-11 10:51:38 +00:00
|
|
|
return 'DROP DATABASE ' . $name;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* getAdvancedForeignKeyOptions
|
|
|
|
* Return the FOREIGN KEY query section dealing with non-standard options
|
|
|
|
* as MATCH, INITIALLY DEFERRED, ON UPDATE, ...
|
|
|
|
*
|
2009-12-02 18:52:21 +00:00
|
|
|
* @param \Doctrine\DBAL\Schema\ForeignKeyConstraint $foreignKey foreign key definition
|
2008-08-31 18:27:16 +00:00
|
|
|
* @return string
|
|
|
|
* @override
|
|
|
|
*/
|
2009-12-02 18:52:21 +00:00
|
|
|
public function getAdvancedForeignKeyOptionsSql(\Doctrine\DBAL\Schema\ForeignKeyConstraint $foreignKey)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
|
|
|
$query = '';
|
2009-12-02 18:52:21 +00:00
|
|
|
if ($foreignKey->hasOption('match')) {
|
|
|
|
$query .= ' MATCH ' . $foreignKey->getOption('match');
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-12-02 18:52:21 +00:00
|
|
|
$query .= parent::getAdvancedForeignKeyOptionsSql($foreignKey);
|
|
|
|
if ($foreignKey->hasOption('deferrable') && $foreignKey->getOption('deferrable') !== false) {
|
2008-08-31 18:27:16 +00:00
|
|
|
$query .= ' DEFERRABLE';
|
|
|
|
} else {
|
|
|
|
$query .= ' NOT DEFERRABLE';
|
|
|
|
}
|
2009-12-02 18:52:21 +00:00
|
|
|
if ($foreignKey->hasOption('feferred') && $foreignKey->getOption('feferred') !== false) {
|
2008-08-31 18:27:16 +00:00
|
|
|
$query .= ' INITIALLY DEFERRED';
|
|
|
|
} else {
|
|
|
|
$query .= ' INITIALLY IMMEDIATE';
|
|
|
|
}
|
|
|
|
return $query;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* generates the sql for altering an existing table on postgresql
|
|
|
|
*
|
|
|
|
* @param string $name name of the table that is intended to be changed.
|
|
|
|
* @param array $changes associative array that contains the details of each type *
|
|
|
|
* @param boolean $check indicates whether the function should just check if the DBMS driver
|
|
|
|
* can perform the requested table alterations if the value is true or
|
|
|
|
* actually perform them otherwise.
|
|
|
|
* @see Doctrine_Export::alterTable()
|
|
|
|
* @return array
|
|
|
|
* @override
|
|
|
|
*/
|
2009-12-06 09:47:54 +00:00
|
|
|
public function getAlterTableSql(TableDiff $diff)
|
|
|
|
{
|
2009-12-06 13:00:49 +00:00
|
|
|
$sql = array();
|
2009-12-06 09:47:54 +00:00
|
|
|
|
|
|
|
foreach ($diff->addedColumns as $column) {
|
|
|
|
$query = 'ADD ' . $this->getColumnDeclarationSql($column->getName(), $column->toArray());
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' ' . $query;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
foreach ($diff->removedColumns as $column) {
|
|
|
|
$query = 'DROP ' . $column->getName();
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' ' . $query;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
foreach ($diff->changedColumns AS $columnDiff) {
|
|
|
|
$oldColumnName = $columnDiff->oldColumnName;
|
|
|
|
$column = $columnDiff->column;
|
|
|
|
|
|
|
|
if ($columnDiff->hasChanged('type')) {
|
|
|
|
$type = $column->getType();
|
2008-08-31 18:27:16 +00:00
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
// here was a server version check before, but DBAL API does not support this anymore.
|
|
|
|
$query = 'ALTER ' . $oldColumnName . ' TYPE ' . $type->getSqlDeclaration($column->toArray(), $this);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' ' . $query;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-12-06 09:47:54 +00:00
|
|
|
if ($columnDiff->hasChanged('default')) {
|
2009-12-06 12:13:15 +00:00
|
|
|
$query = 'ALTER ' . $oldColumnName . ' SET ' . $this->getDefaultValueDeclarationSql($column->toArray());
|
2009-12-06 09:47:54 +00:00
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' ' . $query;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-12-06 09:47:54 +00:00
|
|
|
if ($columnDiff->hasChanged('notnull')) {
|
|
|
|
$query = 'ALTER ' . $oldColumnName . ' ' . ($column->getNotNull() ? 'SET' : 'DROP') . ' NOT NULL';
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' ' . $query;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
foreach ($diff->renamedColumns as $oldColumnName => $column) {
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' RENAME COLUMN ' . $oldColumnName . ' TO ' . $column->getName();
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
2009-12-06 09:47:54 +00:00
|
|
|
if ($diff->newName !== false) {
|
|
|
|
$sql[] = 'ALTER TABLE ' . $diff->name . ' RENAME TO ' . $diff->newName;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
2009-12-06 13:00:49 +00:00
|
|
|
$sql = array_merge($sql, $this->_getAlterTableIndexForeignKeySql($diff));
|
|
|
|
|
2008-08-31 18:27:16 +00:00
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2009-12-02 19:07:38 +00:00
|
|
|
* Gets the SQL to create a sequence on this platform.
|
|
|
|
*
|
|
|
|
* @param \Doctrine\DBAL\Schema\Sequence $sequence
|
2010-02-19 22:33:54 +00:00
|
|
|
* @return string
|
2008-08-31 18:27:16 +00:00
|
|
|
*/
|
2009-12-02 19:07:38 +00:00
|
|
|
public function getCreateSequenceSql(\Doctrine\DBAL\Schema\Sequence $sequence)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
2009-12-02 19:07:38 +00:00
|
|
|
return 'CREATE SEQUENCE ' . $sequence->getName() .
|
|
|
|
' INCREMENT BY ' . $sequence->getAllocationSize() .
|
2009-12-03 13:17:56 +00:00
|
|
|
' MINVALUE ' . $sequence->getInitialValue() .
|
2009-12-02 19:07:38 +00:00
|
|
|
' START ' . $sequence->getInitialValue();
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2009-12-02 22:28:38 +00:00
|
|
|
* Drop existing sequence
|
|
|
|
* @param \Doctrine\DBAL\Schema\Sequence $sequence
|
|
|
|
* @return string
|
2008-08-31 18:27:16 +00:00
|
|
|
*/
|
2009-12-02 22:28:38 +00:00
|
|
|
public function getDropSequenceSql($sequence)
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
2009-12-02 22:28:38 +00:00
|
|
|
if ($sequence instanceof \Doctrine\DBAL\Schema\Sequence) {
|
|
|
|
$sequence = $sequence->getName();
|
|
|
|
}
|
|
|
|
return 'DROP SEQUENCE ' . $sequence;
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
2009-12-06 18:36:46 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param ForeignKeyConstraint|string $foreignKey
|
|
|
|
* @param Table|string $table
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getDropForeignKeySql($foreignKey, $table)
|
|
|
|
{
|
|
|
|
return $this->getDropConstraintSql($foreignKey, $table);
|
|
|
|
}
|
2008-08-31 18:27:16 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the SQL used to create a table.
|
|
|
|
*
|
2009-12-02 18:52:21 +00:00
|
|
|
* @param unknown_type $tableName
|
|
|
|
* @param array $columns
|
2008-08-31 18:27:16 +00:00
|
|
|
* @param array $options
|
|
|
|
* @return unknown
|
|
|
|
*/
|
2009-12-02 18:52:21 +00:00
|
|
|
protected function _getCreateTableSql($tableName, array $columns, array $options = array())
|
2008-08-31 18:27:16 +00:00
|
|
|
{
|
2009-12-02 18:52:21 +00:00
|
|
|
$queryFields = $this->getColumnDeclarationListSql($columns);
|
2008-08-31 18:27:16 +00:00
|
|
|
|
|
|
|
if (isset($options['primary']) && ! empty($options['primary'])) {
|
2009-05-05 17:20:55 +00:00
|
|
|
$keyColumns = array_unique(array_values($options['primary']));
|
2008-08-31 18:27:16 +00:00
|
|
|
$queryFields .= ', PRIMARY KEY(' . implode(', ', $keyColumns) . ')';
|
|
|
|
}
|
|
|
|
|
2009-12-02 18:52:21 +00:00
|
|
|
$query = 'CREATE TABLE ' . $tableName . ' (' . $queryFields . ')';
|
2008-08-31 18:27:16 +00:00
|
|
|
|
|
|
|
$sql[] = $query;
|
|
|
|
|
|
|
|
if (isset($options['indexes']) && ! empty($options['indexes'])) {
|
2009-12-02 18:52:21 +00:00
|
|
|
foreach ($options['indexes'] AS $index) {
|
|
|
|
$sql[] = $this->getCreateIndexSql($index, $tableName);
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($options['foreignKeys'])) {
|
2009-12-02 18:52:21 +00:00
|
|
|
foreach ((array) $options['foreignKeys'] as $definition) {
|
|
|
|
$sql[] = $this->getCreateForeignKeySql($definition, $tableName);
|
2008-08-31 18:27:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Postgres wants boolean values converted to the strings 'true'/'false'.
|
|
|
|
*
|
|
|
|
* @param array $item
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function convertBooleans($item)
|
|
|
|
{
|
|
|
|
if (is_array($item)) {
|
|
|
|
foreach ($item as $key => $value) {
|
|
|
|
if (is_bool($value) || is_numeric($item)) {
|
|
|
|
$item[$key] = ($value) ? 'true' : 'false';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (is_bool($item) || is_numeric($item)) {
|
|
|
|
$item = ($item) ? 'true' : 'false';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $item;
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2008-08-31 18:27:16 +00:00
|
|
|
public function getSequenceNextValSql($sequenceName)
|
|
|
|
{
|
|
|
|
return "SELECT NEXTVAL('" . $sequenceName . "')";
|
|
|
|
}
|
2009-05-28 21:34:35 +00:00
|
|
|
|
2008-09-12 17:25:38 +00:00
|
|
|
public function getSetTransactionIsolationSql($level)
|
|
|
|
{
|
|
|
|
return 'SET SESSION CHARACTERISTICS AS TRANSACTION ISOLATION LEVEL '
|
|
|
|
. $this->_getTransactionIsolationLevelSql($level);
|
|
|
|
}
|
2009-09-05 02:23:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getBooleanTypeDeclarationSql(array $field)
|
|
|
|
{
|
|
|
|
return 'BOOLEAN';
|
|
|
|
}
|
2009-03-28 17:10:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getIntegerTypeDeclarationSql(array $field)
|
|
|
|
{
|
|
|
|
if ( ! empty($field['autoincrement'])) {
|
|
|
|
return 'SERIAL';
|
|
|
|
}
|
2009-09-05 02:23:24 +00:00
|
|
|
|
2009-03-28 17:10:41 +00:00
|
|
|
return 'INT';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getBigIntTypeDeclarationSql(array $field)
|
|
|
|
{
|
|
|
|
if ( ! empty($field['autoincrement'])) {
|
|
|
|
return 'BIGSERIAL';
|
|
|
|
}
|
|
|
|
return 'BIGINT';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getSmallIntTypeDeclarationSql(array $field)
|
|
|
|
{
|
|
|
|
return 'SMALLINT';
|
|
|
|
}
|
|
|
|
|
2009-06-20 12:59:33 +00:00
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getDateTimeTypeDeclarationSql(array $fieldDeclaration)
|
|
|
|
{
|
2009-08-28 17:25:28 +00:00
|
|
|
return 'TIMESTAMP(0) WITH TIME ZONE';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getDateTypeDeclarationSql(array $fieldDeclaration)
|
|
|
|
{
|
|
|
|
return 'DATE';
|
2009-06-20 12:59:33 +00:00
|
|
|
}
|
|
|
|
|
2009-11-03 16:56:05 +00:00
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getTimeTypeDeclarationSql(array $fieldDeclaration)
|
|
|
|
{
|
|
|
|
return 'TIME';
|
|
|
|
}
|
|
|
|
|
2009-03-28 17:10:41 +00:00
|
|
|
/**
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
protected function _getCommonIntegerTypeDeclarationSql(array $columnDef)
|
|
|
|
{
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the SQL snippet used to declare a VARCHAR column on the MySql platform.
|
|
|
|
*
|
|
|
|
* @params array $field
|
|
|
|
* @override
|
|
|
|
*/
|
2009-05-03 18:07:57 +00:00
|
|
|
public function getVarcharTypeDeclarationSql(array $field)
|
2009-03-28 17:10:41 +00:00
|
|
|
{
|
|
|
|
if ( ! isset($field['length'])) {
|
|
|
|
if (array_key_exists('default', $field)) {
|
|
|
|
$field['length'] = $this->getVarcharMaxLength();
|
|
|
|
} else {
|
|
|
|
$field['length'] = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$length = ($field['length'] <= $this->getVarcharMaxLength()) ? $field['length'] : false;
|
|
|
|
$fixed = (isset($field['fixed'])) ? $field['fixed'] : false;
|
|
|
|
|
|
|
|
return $fixed ? ($length ? 'CHAR(' . $length . ')' : 'CHAR(255)')
|
|
|
|
: ($length ? 'VARCHAR(' . $length . ')' : 'TEXT');
|
|
|
|
}
|
2009-10-01 12:00:14 +00:00
|
|
|
|
|
|
|
/** @override */
|
|
|
|
public function getClobTypeDeclarationSql(array $field)
|
|
|
|
{
|
|
|
|
return 'TEXT';
|
|
|
|
}
|
2009-05-27 18:54:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the platform name for this instance
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getName()
|
|
|
|
{
|
|
|
|
return 'postgresql';
|
|
|
|
}
|
2009-08-11 10:51:38 +00:00
|
|
|
|
2009-08-15 18:11:51 +00:00
|
|
|
/**
|
|
|
|
* Gets the character casing of a column in an SQL result set.
|
|
|
|
*
|
|
|
|
* PostgreSQL returns all column names in SQL result sets in lowercase.
|
|
|
|
*
|
|
|
|
* @param string $column The column name for which to get the correct character casing.
|
|
|
|
* @return string The column name in the character casing used in SQL result sets.
|
|
|
|
*/
|
2009-08-11 10:51:38 +00:00
|
|
|
public function getSqlResultCasing($column)
|
|
|
|
{
|
|
|
|
return strtolower($column);
|
|
|
|
}
|
2009-08-28 12:36:06 +00:00
|
|
|
|
2009-08-28 17:25:28 +00:00
|
|
|
public function getDateTimeFormatString()
|
2009-08-28 12:36:06 +00:00
|
|
|
{
|
2009-08-28 17:25:28 +00:00
|
|
|
return 'Y-m-d H:i:sO';
|
|
|
|
}
|
2009-09-08 19:31:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the insert sql for an empty insert statement
|
|
|
|
*
|
|
|
|
* @param string $tableName
|
|
|
|
* @param string $identifierColumnName
|
|
|
|
* @return string $sql
|
|
|
|
*/
|
|
|
|
public function getEmptyIdentityInsertSql($quotedTableName, $quotedIdentifierColumnName)
|
|
|
|
{
|
|
|
|
return 'INSERT INTO ' . $quotedTableName . ' (' . $quotedIdentifierColumnName . ') VALUES (DEFAULT)';
|
|
|
|
}
|
2010-02-07 12:36:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
public function getTruncateTableSql($tableName, $cascade = false)
|
|
|
|
{
|
|
|
|
return 'TRUNCATE '.$tableName.' '.($cascade)?'CASCADE':'';
|
|
|
|
}
|
2009-11-03 16:56:05 +00:00
|
|
|
}
|