2008-08-01 22:46:14 +04:00
|
|
|
<?php
|
|
|
|
|
2009-01-22 22:38:10 +03:00
|
|
|
namespace Doctrine\DBAL\Platforms;
|
2008-08-01 22:46:14 +04:00
|
|
|
|
2009-01-22 22:38:10 +03:00
|
|
|
class PostgreSqlPlatform extends AbstractPlatform
|
2008-08-01 22:46:14 +04:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The reserved keywords by pgsql. Ordered alphabetically.
|
|
|
|
*
|
|
|
|
* @param array
|
|
|
|
* @todo Nedded? What about lazy initialization?
|
|
|
|
*/
|
2008-08-02 21:41:37 +04:00
|
|
|
/*protected static $_reservedKeywords = array(
|
2008-08-01 22:46:14 +04:00
|
|
|
'abort', 'absolute', 'access', 'action', 'add', 'after', 'aggregate',
|
|
|
|
'all', 'alter', 'analyse', 'analyze', 'and', 'any', 'as', 'asc',
|
|
|
|
'assertion', 'assignment', 'at', 'authorization', 'backward', 'before',
|
|
|
|
'begin', 'between', 'bigint', 'binary', 'bit', 'boolean', 'both',
|
|
|
|
'by', 'cache', 'called', 'cascade', 'case', 'cast', 'chain', 'char',
|
|
|
|
'character', 'characteristics', 'check', 'checkpoint', 'class',
|
|
|
|
'close', 'cluster', 'coalesce', 'collate', 'column', 'comment',
|
|
|
|
'commit', 'committed', 'constraint', 'constraints', 'conversion',
|
|
|
|
'convert', 'copy', 'create', 'createdb', 'createuser', 'cross',
|
|
|
|
'current_date', 'current_time', 'current_timestamp', 'current_user',
|
|
|
|
'cursor', 'cycle', 'database', 'day', 'deallocate', 'dec', 'decimal',
|
|
|
|
'declare', 'default', 'deferrable', 'deferred', 'definer', 'delete',
|
|
|
|
'delimiter', 'delimiters', 'desc', 'distinct', 'do', 'domain', 'double',
|
|
|
|
'drop', 'each', 'else', 'encoding', 'encrypted', 'end', 'escape',
|
|
|
|
'except', 'exclusive', 'execute', 'exists', 'explain', 'external',
|
|
|
|
'extract', 'false', 'fetch', 'float', 'for', 'force', 'foreign',
|
|
|
|
'forward', 'freeze', 'from', 'full', 'function', 'get', 'global',
|
|
|
|
'grant', 'group', 'handler', 'having', 'hour', 'ilike', 'immediate',
|
|
|
|
'immutable', 'implicit', 'in', 'increment', 'index', 'inherits',
|
|
|
|
'initially', 'inner', 'inout', 'input', 'insensitive', 'insert',
|
|
|
|
'instead', 'int', 'integer', 'intersect', 'interval', 'into', 'invoker',
|
|
|
|
'is', 'isnull', 'isolation', 'join', 'key', 'lancompiler', 'language',
|
|
|
|
'leading', 'left', 'level', 'like', 'limit', 'listen', 'load', 'local',
|
|
|
|
'localtime', 'localtimestamp', 'location', 'lock', 'match', 'maxvalue',
|
|
|
|
'minute', 'minvalue', 'mode', 'month', 'move', 'names', 'national',
|
|
|
|
'natural', 'nchar', 'new', 'next', 'no', 'nocreatedb', 'nocreateuser',
|
|
|
|
'none', 'not', 'nothing', 'notify', 'notnull', 'null', 'nullif',
|
|
|
|
'numeric', 'of', 'off', 'offset', 'oids', 'old', 'on', 'only', 'operator',
|
|
|
|
'option', 'or', 'order', 'out', 'outer', 'overlaps', 'overlay',
|
|
|
|
'owner', 'partial', 'password', 'path', 'pendant', 'placing', 'position',
|
|
|
|
'precision', 'prepare', 'primary', 'prior', 'privileges', 'procedural',
|
|
|
|
'procedure', 'read', 'real', 'recheck', 'references', 'reindex',
|
|
|
|
'relative', 'rename', 'replace', 'reset', 'restrict', 'returns',
|
|
|
|
'revoke', 'right', 'rollback', 'row', 'rule', 'schema', 'scroll',
|
|
|
|
'second', 'security', 'select', 'sequence', 'serializable', 'session',
|
|
|
|
'session_user', 'set', 'setof', 'share', 'show', 'similar', 'simple',
|
|
|
|
'smallint', 'some', 'stable', 'start', 'statement', 'statistics',
|
|
|
|
'stdin', 'stdout', 'storage', 'strict', 'substring', 'sysid', 'table',
|
|
|
|
'temp', 'template', 'temporary', 'then', 'time', 'timestamp', 'to',
|
|
|
|
'toast', 'trailing', 'transaction', 'treat', 'trigger', 'trim', 'true',
|
|
|
|
'truncate', 'trusted', 'type', 'unencrypted', 'union', 'unique',
|
|
|
|
'unknown', 'unlisten', 'until', 'update', 'usage', 'user', 'using',
|
|
|
|
'vacuum', 'valid', 'validator', 'values', 'varchar', 'varying',
|
|
|
|
'verbose', 'version', 'view', 'volatile', 'when', 'where', 'with',
|
2008-08-02 21:41:37 +04:00
|
|
|
'without', 'work', 'write', 'year','zone');*/
|
2008-08-01 22:46:14 +04:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor.
|
|
|
|
* Creates a new PostgreSqlPlatform.
|
|
|
|
*/
|
|
|
|
public function __construct()
|
|
|
|
{
|
2008-08-02 21:41:37 +04:00
|
|
|
parent::__construct();
|
2008-08-01 22:46:14 +04:00
|
|
|
$this->_properties['string_quoting'] = array('start' => "'",
|
|
|
|
'end' => "'",
|
|
|
|
'escape' => "'",
|
|
|
|
'escape_pattern' => '\\');
|
|
|
|
$this->_properties['identifier_quoting'] = array('start' => '"',
|
|
|
|
'end' => '"',
|
|
|
|
'escape' => '"');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Obtain DBMS specific SQL code portion needed to declare an text type
|
|
|
|
* field to be used in statements like CREATE TABLE.
|
|
|
|
*
|
|
|
|
* @param array $field associative array with the name of the properties
|
|
|
|
* of the field being declared as array indexes. Currently, the types
|
|
|
|
* of supported field properties are as follows:
|
|
|
|
*
|
|
|
|
* length
|
|
|
|
* Integer value that determines the maximum length of the text
|
|
|
|
* field. If this argument is missing the field should be
|
|
|
|
* declared to have the longest length allowed by the DBMS.
|
|
|
|
*
|
|
|
|
* default
|
|
|
|
* Text value to be used as default for this field.
|
|
|
|
*
|
|
|
|
* notnull
|
|
|
|
* Boolean flag that indicates whether this field is constrained
|
|
|
|
* to not be set to null.
|
|
|
|
*
|
|
|
|
* @return string DBMS specific SQL code portion that should be used to
|
|
|
|
* declare the specified field.
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getNativeDeclaration(array $field)
|
|
|
|
{
|
|
|
|
if ( ! isset($field['type'])) {
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('Missing column type.');
|
2008-08-01 22:46:14 +04:00
|
|
|
}
|
|
|
|
switch ($field['type']) {
|
|
|
|
case 'char':
|
|
|
|
case 'string':
|
|
|
|
case 'array':
|
|
|
|
case 'object':
|
|
|
|
case 'varchar':
|
|
|
|
case 'gzip':
|
|
|
|
// TODO: what is the maximum VARCHAR length in pgsql ?
|
|
|
|
$length = (isset($field['length']) && $field['length'] && $field['length'] < 10000) ? $field['length'] : null;
|
|
|
|
|
|
|
|
$fixed = ((isset($field['fixed']) && $field['fixed']) || $field['type'] == 'char') ? true : false;
|
|
|
|
|
|
|
|
return $fixed ? ($length ? 'CHAR(' . $length . ')' : 'CHAR('.$this->conn->options['default_text_field_length'].')')
|
|
|
|
: ($length ? 'VARCHAR(' .$length . ')' : 'TEXT');
|
|
|
|
|
|
|
|
case 'clob':
|
|
|
|
return 'TEXT';
|
|
|
|
case 'blob':
|
|
|
|
return 'BYTEA';
|
|
|
|
case 'enum':
|
|
|
|
case 'integer':
|
|
|
|
case 'int':
|
|
|
|
if ( ! empty($field['autoincrement'])) {
|
|
|
|
if ( ! empty($field['length'])) {
|
|
|
|
$length = $field['length'];
|
|
|
|
if ($length > 4) {
|
|
|
|
return 'BIGSERIAL';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 'SERIAL';
|
|
|
|
}
|
|
|
|
if ( ! empty($field['length'])) {
|
|
|
|
$length = $field['length'];
|
|
|
|
if ($length <= 2) {
|
|
|
|
return 'SMALLINT';
|
|
|
|
} elseif ($length == 3 || $length == 4) {
|
|
|
|
return 'INT';
|
|
|
|
} elseif ($length > 4) {
|
|
|
|
return 'BIGINT';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 'INT';
|
|
|
|
case 'boolean':
|
|
|
|
return 'BOOLEAN';
|
|
|
|
case 'date':
|
|
|
|
return 'DATE';
|
|
|
|
case 'time':
|
|
|
|
return 'TIME without time zone';
|
|
|
|
case 'timestamp':
|
|
|
|
return 'TIMESTAMP without time zone';
|
|
|
|
case 'float':
|
|
|
|
case 'double':
|
|
|
|
return 'FLOAT';
|
|
|
|
case 'decimal':
|
|
|
|
$length = !empty($field['length']) ? $field['length'] : 18;
|
|
|
|
$scale = !empty($field['scale']) ? $field['scale'] : $this->conn->getAttribute(Doctrine::ATTR_DECIMAL_PLACES);
|
|
|
|
return 'NUMERIC('.$length.','.$scale.')';
|
|
|
|
}
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('Unknown field type \'' . $field['type'] . '\'.');
|
2008-08-01 22:46:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps a native array description of a field to a portable Doctrine datatype and length
|
|
|
|
*
|
|
|
|
* @param array $field native field description
|
|
|
|
*
|
|
|
|
* @return array containing the various possible types, length, sign, fixed
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getPortableDeclaration(array $field)
|
|
|
|
{
|
|
|
|
|
|
|
|
$length = (isset($field['length'])) ? $field['length'] : null;
|
|
|
|
if ($length == '-1' && isset($field['atttypmod'])) {
|
|
|
|
$length = $field['atttypmod'] - 4;
|
|
|
|
}
|
|
|
|
if ((int)$length <= 0) {
|
|
|
|
$length = null;
|
|
|
|
}
|
|
|
|
$type = array();
|
|
|
|
$unsigned = $fixed = null;
|
|
|
|
|
|
|
|
if ( ! isset($field['name'])) {
|
|
|
|
$field['name'] = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
$dbType = strtolower($field['type']);
|
|
|
|
|
|
|
|
switch ($dbType) {
|
|
|
|
case 'smallint':
|
|
|
|
case 'int2':
|
|
|
|
$type[] = 'integer';
|
|
|
|
$unsigned = false;
|
|
|
|
$length = 2;
|
|
|
|
if ($length == '2') {
|
|
|
|
$type[] = 'boolean';
|
|
|
|
if (preg_match('/^(is|has)/', $field['name'])) {
|
|
|
|
$type = array_reverse($type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'int':
|
|
|
|
case 'int4':
|
|
|
|
case 'integer':
|
|
|
|
case 'serial':
|
|
|
|
case 'serial4':
|
|
|
|
$type[] = 'integer';
|
|
|
|
$unsigned = false;
|
|
|
|
$length = 4;
|
|
|
|
break;
|
|
|
|
case 'bigint':
|
|
|
|
case 'int8':
|
|
|
|
case 'bigserial':
|
|
|
|
case 'serial8':
|
|
|
|
$type[] = 'integer';
|
|
|
|
$unsigned = false;
|
|
|
|
$length = 8;
|
|
|
|
break;
|
|
|
|
case 'bool':
|
|
|
|
case 'boolean':
|
|
|
|
$type[] = 'boolean';
|
|
|
|
$length = 1;
|
|
|
|
break;
|
|
|
|
case 'text':
|
|
|
|
case 'varchar':
|
|
|
|
case 'interval':
|
|
|
|
case '_varchar':
|
|
|
|
$fixed = false;
|
|
|
|
case 'unknown':
|
|
|
|
case 'char':
|
|
|
|
case 'bpchar':
|
|
|
|
$type[] = 'string';
|
|
|
|
if ($length == '1') {
|
|
|
|
$type[] = 'boolean';
|
|
|
|
if (preg_match('/^(is|has)/', $field['name'])) {
|
|
|
|
$type = array_reverse($type);
|
|
|
|
}
|
|
|
|
} elseif (strstr($dbType, 'text')) {
|
|
|
|
$type[] = 'clob';
|
|
|
|
}
|
|
|
|
if ($fixed !== false) {
|
|
|
|
$fixed = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'date':
|
|
|
|
$type[] = 'date';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
case 'datetime':
|
|
|
|
case 'timestamp':
|
|
|
|
case 'timestamptz':
|
|
|
|
$type[] = 'timestamp';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
case 'time':
|
|
|
|
$type[] = 'time';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
case 'float':
|
|
|
|
case 'float4':
|
|
|
|
case 'float8':
|
|
|
|
case 'double':
|
|
|
|
case 'double precision':
|
|
|
|
case 'real':
|
|
|
|
$type[] = 'float';
|
|
|
|
break;
|
|
|
|
case 'decimal':
|
|
|
|
case 'money':
|
|
|
|
case 'numeric':
|
|
|
|
$type[] = 'decimal';
|
|
|
|
break;
|
|
|
|
case 'tinyblob':
|
|
|
|
case 'mediumblob':
|
|
|
|
case 'longblob':
|
|
|
|
case 'blob':
|
|
|
|
case 'bytea':
|
|
|
|
$type[] = 'blob';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
case 'oid':
|
|
|
|
$type[] = 'blob';
|
|
|
|
$type[] = 'clob';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
case 'year':
|
|
|
|
$type[] = 'integer';
|
|
|
|
$type[] = 'date';
|
|
|
|
$length = null;
|
|
|
|
break;
|
|
|
|
default:
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('unknown database attribute type: '.$dbType);
|
2008-08-01 22:46:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return array('type' => $type,
|
|
|
|
'length' => $length,
|
|
|
|
'unsigned' => $unsigned,
|
|
|
|
'fixed' => $fixed);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the md5 sum of a field.
|
|
|
|
*
|
|
|
|
* Note: Not SQL92, but common functionality
|
|
|
|
*
|
|
|
|
* md5() works with the default PostgreSQL 8 versions.
|
|
|
|
*
|
|
|
|
* If you are using PostgreSQL 7.x or older you need
|
|
|
|
* to make sure that the digest procedure is installed.
|
|
|
|
* If you use RPMS (Redhat and Mandrake) install the postgresql-contrib
|
|
|
|
* package. You must then install the procedure by running this shell command:
|
|
|
|
* <code>
|
|
|
|
* psql [dbname] < /usr/share/pgsql/contrib/pgcrypto.sql
|
|
|
|
* </code>
|
|
|
|
* You should make sure you run this as the postgres user.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getMd5Expression($column)
|
|
|
|
{
|
|
|
|
$column = $this->getIdentifier($column);
|
|
|
|
|
|
|
|
if ($this->_version > 7) {
|
|
|
|
return 'MD5(' . $column . ')';
|
|
|
|
} else {
|
|
|
|
return 'encode(digest(' . $column .', md5), hex)';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
|
|
|
$value = $this->getIdentifier($value);
|
|
|
|
|
|
|
|
if ($len === null) {
|
|
|
|
$len = $this->getIdentifier($len);
|
|
|
|
return 'SUBSTR(' . $value . ', ' . $from . ')';
|
|
|
|
} else {
|
|
|
|
return 'SUBSTR(' . $value . ', ' . $from . ', ' . $len . ')';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PostgreSQLs AGE(<timestamp1> [, <timestamp2>]) function.
|
|
|
|
*
|
|
|
|
* @param string $timestamp1 timestamp to subtract from NOW()
|
|
|
|
* @param string $timestamp2 optional; if given: subtract arguments
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getAgeExpression($timestamp1, $timestamp2 = null)
|
|
|
|
{
|
|
|
|
if ( $timestamp2 == null ) {
|
|
|
|
return 'AGE(' . $timestamp1 . ')';
|
|
|
|
}
|
|
|
|
return 'AGE(' . $timestamp1 . ', ' . $timestamp2 . ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PostgreSQLs DATE_PART( <text>, <time> ) function.
|
|
|
|
*
|
|
|
|
* @param string $text what to extract
|
|
|
|
* @param string $time timestamp or interval to extract from
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getDatePartExpression($text, $time)
|
|
|
|
{
|
|
|
|
return 'DATE_PART(' . $text . ', ' . $time . ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PostgreSQLs TO_CHAR( <time>, <text> ) function.
|
|
|
|
*
|
|
|
|
* @param string $time timestamp or interval
|
|
|
|
* @param string $text how to the format the output
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getToCharExpression($time, $text)
|
|
|
|
{
|
|
|
|
return 'TO_CHAR(' . $time . ', ' . $text . ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* return string to call a function to get random value inside an SQL statement
|
|
|
|
*
|
|
|
|
* @return return string to generate float between 0 and 1
|
|
|
|
* @access public
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getRandomExpression()
|
|
|
|
{
|
|
|
|
return 'RANDOM()';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* build a pattern matching string
|
|
|
|
*
|
|
|
|
* EXPERIMENTAL
|
|
|
|
*
|
|
|
|
* WARNING: this function is experimental and may change signature at
|
|
|
|
* any time until labelled as non-experimental
|
|
|
|
*
|
|
|
|
* @access public
|
|
|
|
*
|
|
|
|
* @param array $pattern even keys are strings, odd are patterns (% and _)
|
|
|
|
* @param string $operator optional pattern operator (LIKE, ILIKE and maybe others in the future)
|
|
|
|
* @param string $field optional field name that is being matched against
|
|
|
|
* (might be required when emulating ILIKE)
|
|
|
|
*
|
|
|
|
* @return string SQL pattern
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getMatchPatternExpression($pattern, $operator = null, $field = null)
|
|
|
|
{
|
|
|
|
$match = '';
|
|
|
|
if ( ! is_null($operator)) {
|
|
|
|
$field = is_null($field) ? '' : $field.' ';
|
|
|
|
$operator = strtoupper($operator);
|
|
|
|
switch ($operator) {
|
|
|
|
// case insensitive
|
|
|
|
case 'ILIKE':
|
|
|
|
$match = $field.'ILIKE ';
|
|
|
|
break;
|
|
|
|
// case sensitive
|
|
|
|
case 'LIKE':
|
|
|
|
$match = $field.'LIKE ';
|
|
|
|
break;
|
|
|
|
default:
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('not a supported operator type:'. $operator);
|
2008-08-01 22:46:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
$match.= "'";
|
|
|
|
foreach ($pattern as $key => $value) {
|
|
|
|
if ($key % 2) {
|
|
|
|
$match.= $value;
|
|
|
|
} else {
|
|
|
|
$match.= $this->conn->escapePattern($this->conn->escape($value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$match.= "'";
|
|
|
|
$match.= $this->patternEscapeString();
|
|
|
|
|
|
|
|
return $match;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* parses a literal boolean value and returns
|
|
|
|
* proper sql equivalent
|
|
|
|
*
|
|
|
|
* @param string $value boolean value to be parsed
|
|
|
|
* @return string parsed boolean value
|
|
|
|
*/
|
|
|
|
public function parseBoolean($value)
|
|
|
|
{
|
|
|
|
return $value;
|
2008-08-02 21:41:37 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the platform supports sequences.
|
|
|
|
* Postgres has native support for sequences.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function supportsSequences()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
2008-08-31 22:27:16 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListDatabasesSql()
|
|
|
|
{
|
|
|
|
return 'SELECT datname FROM pg_database';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListFunctionsSql()
|
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
proname
|
|
|
|
FROM
|
|
|
|
pg_proc pr, pg_type tp
|
|
|
|
WHERE
|
|
|
|
tp.oid = pr.prorettype
|
|
|
|
AND pr.proisagg = FALSE
|
|
|
|
AND tp.typname <> 'trigger'
|
|
|
|
AND pr.pronamespace IN
|
|
|
|
(SELECT oid FROM pg_namespace
|
|
|
|
WHERE nspname NOT LIKE 'pg_%' AND nspname != 'information_schema'";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListSequencesSql()
|
|
|
|
{
|
|
|
|
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')";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
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_'";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListViewsSql()
|
|
|
|
{
|
|
|
|
return 'SELECT viewname FROM pg_views';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListUsersSql()
|
|
|
|
{
|
|
|
|
return 'SELECT usename FROM pg_user';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListTableConstraintsSql()
|
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
relname
|
|
|
|
FROM
|
|
|
|
pg_class
|
|
|
|
WHERE oid IN (
|
|
|
|
SELECT indexrelid
|
|
|
|
FROM pg_index, pg_class
|
|
|
|
WHERE pg_class.relname = %s
|
|
|
|
AND pg_class.oid = pg_index.indrelid
|
|
|
|
AND (indisunique = 't' OR indisprimary = 't')
|
|
|
|
)";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListTableIndexesSql()
|
|
|
|
{
|
|
|
|
return "SELECT
|
|
|
|
relname
|
|
|
|
FROM
|
|
|
|
pg_class
|
|
|
|
WHERE oid IN (
|
|
|
|
SELECT indexrelid
|
|
|
|
FROM pg_index, pg_class
|
|
|
|
WHERE pg_class.relname = %s
|
|
|
|
AND pg_class.oid=pg_index.indrelid
|
|
|
|
AND indisunique != 't'
|
|
|
|
AND indisprimary != 't'
|
|
|
|
)";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getListTableColumnsSql()
|
|
|
|
{
|
|
|
|
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
|
|
|
|
WHERE c.relname = %s
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
return 'CREATE DATABASE ' . $this->quoteIdentifier($name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drop an existing database
|
|
|
|
*
|
|
|
|
* @param string $name name of the database that should be dropped
|
|
|
|
* @throws PDOException
|
|
|
|
* @access public
|
|
|
|
*/
|
|
|
|
public function getDropDatabaseSql($name)
|
|
|
|
{
|
|
|
|
return 'DROP DATABASE ' . $this->quoteIdentifier($name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* getAdvancedForeignKeyOptions
|
|
|
|
* Return the FOREIGN KEY query section dealing with non-standard options
|
|
|
|
* as MATCH, INITIALLY DEFERRED, ON UPDATE, ...
|
|
|
|
*
|
|
|
|
* @param array $definition foreign key definition
|
|
|
|
* @return string
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getAdvancedForeignKeyOptionsSql(array $definition)
|
|
|
|
{
|
|
|
|
$query = '';
|
|
|
|
if (isset($definition['match'])) {
|
|
|
|
$query .= ' MATCH ' . $definition['match'];
|
|
|
|
}
|
|
|
|
if (isset($definition['onUpdate'])) {
|
|
|
|
$query .= ' ON UPDATE ' . $definition['onUpdate'];
|
|
|
|
}
|
|
|
|
if (isset($definition['onDelete'])) {
|
|
|
|
$query .= ' ON DELETE ' . $definition['onDelete'];
|
|
|
|
}
|
|
|
|
if (isset($definition['deferrable'])) {
|
|
|
|
$query .= ' DEFERRABLE';
|
|
|
|
} else {
|
|
|
|
$query .= ' NOT DEFERRABLE';
|
|
|
|
}
|
|
|
|
if (isset($definition['feferred'])) {
|
|
|
|
$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
|
|
|
|
*/
|
|
|
|
public function getAlterTableSql($name, array $changes, $check = false)
|
|
|
|
{
|
|
|
|
foreach ($changes as $changeName => $change) {
|
|
|
|
switch ($changeName) {
|
|
|
|
case 'add':
|
|
|
|
case 'remove':
|
|
|
|
case 'change':
|
|
|
|
case 'name':
|
|
|
|
case 'rename':
|
|
|
|
break;
|
|
|
|
default:
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('change type "' . $changeName . '\" not yet supported');
|
2008-08-31 22:27:16 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($check) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql = array();
|
|
|
|
|
|
|
|
if (isset($changes['add']) && is_array($changes['add'])) {
|
|
|
|
foreach ($changes['add'] as $fieldName => $field) {
|
|
|
|
$query = 'ADD ' . $this->getDeclarationSql($fieldName, $field);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($changes['remove']) && is_array($changes['remove'])) {
|
|
|
|
foreach ($changes['remove'] as $fieldName => $field) {
|
|
|
|
$fieldName = $this->quoteIdentifier($fieldName, true);
|
|
|
|
$query = 'DROP ' . $fieldName;
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($changes['change']) && is_array($changes['change'])) {
|
|
|
|
foreach ($changes['change'] as $fieldName => $field) {
|
|
|
|
$fieldName = $this->quoteIdentifier($fieldName, true);
|
|
|
|
if (isset($field['type'])) {
|
|
|
|
$serverInfo = $this->getServerVersion();
|
|
|
|
|
|
|
|
if (is_array($serverInfo) && $serverInfo['major'] < 8) {
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('changing column type for "'.$field['type'].'\" requires PostgreSQL 8.0 or above');
|
2008-08-31 22:27:16 +04:00
|
|
|
}
|
|
|
|
$query = 'ALTER ' . $fieldName . ' TYPE ' . $this->getTypeDeclarationSql($field['definition']);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
|
|
|
|
}
|
|
|
|
if (array_key_exists('default', $field)) {
|
|
|
|
$query = 'ALTER ' . $fieldName . ' SET DEFAULT ' . $this->quote($field['definition']['default'], $field['definition']['type']);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
|
|
|
|
}
|
|
|
|
if ( ! empty($field['notnull'])) {
|
|
|
|
$query = 'ALTER ' . $fieldName . ' ' . ($field['definition']['notnull'] ? 'SET' : 'DROP') . ' NOT NULL';
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($changes['rename']) && is_array($changes['rename'])) {
|
|
|
|
foreach ($changes['rename'] as $fieldName => $field) {
|
|
|
|
$fieldName = $this->quoteIdentifier($fieldName, true);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' RENAME COLUMN ' . $fieldName . ' TO ' . $this->quoteIdentifier($field['name'], true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$name = $this->quoteIdentifier($name, true);
|
|
|
|
if (isset($changes['name'])) {
|
|
|
|
$changeName = $this->quoteIdentifier($changes['name'], true);
|
|
|
|
$sql[] = 'ALTER TABLE ' . $name . ' RENAME TO ' . $changeName;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* return RDBMS specific create sequence statement
|
|
|
|
*
|
2009-02-20 08:46:20 +03:00
|
|
|
* @throws Doctrine\DBAL\ConnectionException if something fails at database level
|
2008-08-31 22:27:16 +04:00
|
|
|
* @param string $seqName name of the sequence to be created
|
|
|
|
* @param string $start start value of the sequence; default is 1
|
|
|
|
* @param array $options An associative array of table options:
|
|
|
|
* array(
|
|
|
|
* 'comment' => 'Foo',
|
|
|
|
* 'charset' => 'utf8',
|
|
|
|
* 'collate' => 'utf8_unicode_ci',
|
|
|
|
* );
|
|
|
|
* @return string
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getCreateSequenceSql($sequenceName, $start = 1, array $options = array())
|
|
|
|
{
|
|
|
|
$sequenceName = $this->quoteIdentifier($this->formatter->getSequenceName($sequenceName), true);
|
|
|
|
return 'CREATE SEQUENCE ' . $sequenceName . ' INCREMENT 1' .
|
|
|
|
($start < 1 ? ' MINVALUE ' . $start : '') . ' START ' . $start;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drop existing sequence
|
|
|
|
*
|
|
|
|
* @param string $sequenceName name of the sequence to be dropped
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getDropSequenceSql($sequenceName)
|
|
|
|
{
|
|
|
|
$sequenceName = $this->quoteIdentifier($this->formatter->getSequenceName($sequenceName), true);
|
|
|
|
return 'DROP SEQUENCE ' . $sequenceName;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the SQL used to create a table.
|
|
|
|
*
|
|
|
|
* @param unknown_type $name
|
|
|
|
* @param array $fields
|
|
|
|
* @param array $options
|
|
|
|
* @return unknown
|
|
|
|
*/
|
|
|
|
public function getCreateTableSql($name, array $fields, array $options = array())
|
|
|
|
{
|
|
|
|
if ( ! $name) {
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('no valid table name specified');
|
2008-08-31 22:27:16 +04:00
|
|
|
}
|
|
|
|
if (empty($fields)) {
|
2009-02-19 10:00:54 +03:00
|
|
|
throw \Doctrine\Common\DoctrineException::updateMe('no fields specified for table ' . $name);
|
2008-08-31 22:27:16 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
$queryFields = $this->getFieldDeclarationListSql($fields);
|
|
|
|
|
|
|
|
if (isset($options['primary']) && ! empty($options['primary'])) {
|
|
|
|
$keyColumns = array_values($options['primary']);
|
|
|
|
$keyColumns = array_map(array($this, 'quoteIdentifier'), $keyColumns);
|
|
|
|
$queryFields .= ', PRIMARY KEY(' . implode(', ', $keyColumns) . ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
$query = 'CREATE TABLE ' . $this->quoteIdentifier($name, true) . ' (' . $queryFields . ')';
|
|
|
|
|
|
|
|
$sql[] = $query;
|
|
|
|
|
|
|
|
if (isset($options['indexes']) && ! empty($options['indexes'])) {
|
|
|
|
foreach($options['indexes'] as $index => $definition) {
|
|
|
|
$sql[] = $this->getCreateIndexSql($name, $index, $definition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($options['foreignKeys'])) {
|
|
|
|
|
|
|
|
foreach ((array) $options['foreignKeys'] as $k => $definition) {
|
|
|
|
if (is_array($definition)) {
|
|
|
|
$sql[] = $this->getCreateForeignKeySql($name, $definition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Obtain DBMS specific SQL code portion needed to declare an integer type
|
|
|
|
* field to be used in statements like CREATE TABLE.
|
|
|
|
*
|
|
|
|
* @param string $name name the field to be declared.
|
|
|
|
* @param array $field associative array with the name of the properties
|
|
|
|
* of the field being declared as array indexes. Currently, the types
|
|
|
|
* of supported field properties are as follows:
|
|
|
|
*
|
|
|
|
* unsigned
|
|
|
|
* Boolean flag that indicates whether the field should be
|
|
|
|
* declared as unsigned integer if possible.
|
|
|
|
*
|
|
|
|
* default
|
|
|
|
* Integer value to be used as default for this field.
|
|
|
|
*
|
|
|
|
* notnull
|
|
|
|
* Boolean flag that indicates whether this field is constrained
|
|
|
|
* to not be set to null.
|
|
|
|
* @return string DBMS specific SQL code portion that should be used to
|
|
|
|
* declare the specified field.
|
|
|
|
*/
|
|
|
|
public function getIntegerDeclarationSql($name, $field)
|
|
|
|
{
|
|
|
|
if ( ! empty($field['autoincrement'])) {
|
|
|
|
$name = $this->quoteIdentifier($name, true);
|
|
|
|
return $name . ' ' . $this->getNativeDeclaration($field);
|
|
|
|
}
|
|
|
|
|
|
|
|
$default = '';
|
|
|
|
if (array_key_exists('default', $field)) {
|
|
|
|
if ($field['default'] === '') {
|
|
|
|
$field['default'] = empty($field['notnull']) ? null : 0;
|
|
|
|
}
|
|
|
|
$default = ' DEFAULT '.$this->quote($field['default'], $field['type']);
|
|
|
|
} elseif (empty($field['notnull'])) {
|
|
|
|
$default = ' DEFAULT NULL';
|
|
|
|
}
|
|
|
|
|
|
|
|
$notnull = empty($field['notnull']) ? '' : ' NOT NULL';
|
|
|
|
$name = $this->quoteIdentifier($name, true);
|
|
|
|
|
|
|
|
return $name . ' ' . $this->getNativeDeclaration($field) . $default . $notnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Postgres wants boolean values converted to the strings 'true'/'false'.
|
|
|
|
*
|
|
|
|
* @param array $item
|
|
|
|
* @return void
|
|
|
|
* @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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @param string $sequenceName
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getSequenceNextValSql($sequenceName)
|
|
|
|
{
|
|
|
|
return "SELECT NEXTVAL('" . $sequenceName . "')";
|
|
|
|
}
|
2008-09-12 21:25:38 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Enter description here...
|
|
|
|
*
|
|
|
|
* @param unknown_type $level
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
public function getSetTransactionIsolationSql($level)
|
|
|
|
{
|
|
|
|
return 'SET SESSION CHARACTERISTICS AS TRANSACTION ISOLATION LEVEL '
|
|
|
|
. $this->_getTransactionIsolationLevelSql($level);
|
|
|
|
}
|
2009-02-20 08:46:20 +03:00
|
|
|
}
|