Вход Регистрация
Файл: vendor/doctrine/dbal/lib/Doctrine/DBAL/Platforms/OraclePlatform.php
Строк: 1451
<?php

namespace DoctrineDBALPlatforms;

use 
DoctrineDBALException;
use 
DoctrineDBALSchemaForeignKeyConstraint;
use 
DoctrineDBALSchemaIdentifier;
use 
DoctrineDBALSchemaIndex;
use 
DoctrineDBALSchemaSequence;
use 
DoctrineDBALSchemaTable;
use 
DoctrineDBALSchemaTableDiff;
use 
DoctrineDBALTransactionIsolationLevel;
use 
DoctrineDBALTypesBinaryType;
use 
DoctrineDeprecationsDeprecation;
use 
InvalidArgumentException;

use function 
array_merge;
use function 
count;
use function 
explode;
use function 
func_get_arg;
use function 
func_num_args;
use function 
implode;
use function 
preg_match;
use function 
sprintf;
use function 
strlen;
use function 
strpos;
use function 
strtoupper;
use function 
substr;

/**
 * OraclePlatform.
 */
class OraclePlatform extends AbstractPlatform
{
    
/**
     * Assertion for Oracle identifiers.
     *
     * @link http://docs.oracle.com/cd/B19306_01/server.102/b14200/sql_elements008.htm
     *
     * @param string $identifier
     *
     * @return void
     *
     * @throws Exception
     */
    
public static function assertValidIdentifier($identifier)
    {
        if (! 
preg_match('(^(([a-zA-Z]{1}[a-zA-Z0-9_$#]{0,})|("[^"]+"))$)'$identifier)) {
            throw new 
Exception('Invalid Oracle identifier');
        }
    }

    
/**
     * {@inheritDoc}
     */
    
public function getSubstringExpression($string$start$length null)
    {
        if (
$length !== null) {
            return 
sprintf('SUBSTR(%s, %d, %d)'$string$start$length);
        }

        return 
sprintf('SUBSTR(%s, %d)'$string$start);
    }

    
/**
     * @param string $type
     *
     * @return string
     */
    
public function getNowExpression($type 'timestamp')
    {
        switch (
$type) {
            case 
'date':
            case 
'time':
            case 
'timestamp':
            default:
                return 
'TO_CHAR(CURRENT_TIMESTAMP, 'YYYY-MM-DD HH24:MI:SS')';
        }
    }

    
/**
     * {@inheritDoc}
     */
    
public function getLocateExpression($str$substr$startPos false)
    {
        if (
$startPos === false) {
            return 
'INSTR(' $str ', ' $substr ')';
        }

        return 
'INSTR(' $str ', ' $substr ', ' $startPos ')';
    }

    
/**
     * {@inheritDoc}
     *
     * @deprecated Use application-generated UUIDs instead
     */
    
public function getGuidExpression()
    {
        return 
'SYS_GUID()';
    }

    
/**
     * {@inheritdoc}
     */
    
protected function getDateArithmeticIntervalExpression($date$operator$interval$unit)
    {
        switch (
$unit) {
            case 
DateIntervalUnit::MONTH:
            case 
DateIntervalUnit::QUARTER:
            case 
DateIntervalUnit::YEAR:
                switch (
$unit) {
                    case 
DateIntervalUnit::QUARTER:
                        
$interval *= 3;
                        break;

                    case 
DateIntervalUnit::YEAR:
                        
$interval *= 12;
                        break;
                }

                return 
'ADD_MONTHS(' $date ', ' $operator $interval ')';

            default:
                
$calculationClause '';

                switch (
$unit) {
                    case 
DateIntervalUnit::SECOND:
                        
$calculationClause '/24/60/60';
                        break;

                    case 
DateIntervalUnit::MINUTE:
                        
$calculationClause '/24/60';
                        break;

                    case 
DateIntervalUnit::HOUR:
                        
$calculationClause '/24';
                        break;

                    case 
DateIntervalUnit::WEEK:
                        
$calculationClause '*7';
                        break;
                }

                return 
'(' $date $operator $interval $calculationClause ')';
        }
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateDiffExpression($date1$date2)
    {
        return 
sprintf('TRUNC(%s) - TRUNC(%s)'$date1$date2);
    }

    
/**
     * {@inheritDoc}
     */
    
public function getBitAndComparisonExpression($value1$value2)
    {
        return 
'BITAND(' $value1 ', ' $value2 ')';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getBitOrComparisonExpression($value1$value2)
    {
        return 
'(' $value1 '-' .
                
$this->getBitAndComparisonExpression($value1$value2)
                . 
'+' $value2 ')';
    }

    
/**
     * {@inheritDoc}
     *
     * Need to specifiy minvalue, since start with is hidden in the system and MINVALUE <= START WITH.
     * Therefore we can use MINVALUE to be able to get a hint what START WITH was for later introspection
     * in {@see listSequences()}
     */
    
public function getCreateSequenceSQL(Sequence $sequence)
    {
        return 
'CREATE SEQUENCE ' $sequence->getQuotedName($this) .
               
' START WITH ' $sequence->getInitialValue() .
               
' MINVALUE ' $sequence->getInitialValue() .
               
' INCREMENT BY ' $sequence->getAllocationSize() .
               
$this->getSequenceCacheSQL($sequence);
    }

    
/**
     * {@inheritDoc}
     */
    
public function getAlterSequenceSQL(Sequence $sequence)
    {
        return 
'ALTER SEQUENCE ' $sequence->getQuotedName($this) .
               
' INCREMENT BY ' $sequence->getAllocationSize()
               . 
$this->getSequenceCacheSQL($sequence);
    }

    
/**
     * Cache definition for sequences
     *
     * @return string
     */
    
private function getSequenceCacheSQL(Sequence $sequence)
    {
        if (
$sequence->getCache() === 0) {
            return 
' NOCACHE';
        }

        if (
$sequence->getCache() === 1) {
            return 
' NOCACHE';
        }

        if (
$sequence->getCache() > 1) {
            return 
' CACHE ' $sequence->getCache();
        }

        return 
'';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getSequenceNextValSQL($sequence)
    {
        return 
'SELECT ' $sequence '.nextval FROM DUAL';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getSetTransactionIsolationSQL($level)
    {
        return 
'SET TRANSACTION ISOLATION LEVEL ' $this->_getTransactionIsolationLevelSQL($level);
    }

    
/**
     * {@inheritDoc}
     */
    
protected function _getTransactionIsolationLevelSQL($level)
    {
        switch (
$level) {
            case 
TransactionIsolationLevel::READ_UNCOMMITTED:
                return 
'READ UNCOMMITTED';

            case 
TransactionIsolationLevel::READ_COMMITTED:
                return 
'READ COMMITTED';

            case 
TransactionIsolationLevel::REPEATABLE_READ:
            case 
TransactionIsolationLevel::SERIALIZABLE:
                return 
'SERIALIZABLE';

            default:
                return 
parent::_getTransactionIsolationLevelSQL($level);
        }
    }

    
/**
     * {@inheritDoc}
     */
    
public function getBooleanTypeDeclarationSQL(array $column)
    {
        return 
'NUMBER(1)';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getIntegerTypeDeclarationSQL(array $column)
    {
        return 
'NUMBER(10)';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getBigIntTypeDeclarationSQL(array $column)
    {
        return 
'NUMBER(20)';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getSmallIntTypeDeclarationSQL(array $column)
    {
        return 
'NUMBER(5)';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateTimeTypeDeclarationSQL(array $column)
    {
        return 
'TIMESTAMP(0)';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateTimeTzTypeDeclarationSQL(array $column)
    {
        return 
'TIMESTAMP(0) WITH TIME ZONE';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateTypeDeclarationSQL(array $column)
    {
        return 
'DATE';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getTimeTypeDeclarationSQL(array $column)
    {
        return 
'DATE';
    }

    
/**
     * {@inheritDoc}
     */
    
protected function _getCommonIntegerTypeDeclarationSQL(array $column)
    {
        return 
'';
    }

    
/**
     * {@inheritDoc}
     */
    
protected function getVarcharTypeDeclarationSQLSnippet($length$fixed)
    {
        return 
$fixed ? ($length 'CHAR(' $length ')' 'CHAR(2000)')
                : (
$length 'VARCHAR2(' $length ')' 'VARCHAR2(4000)');
    }

    
/**
     * {@inheritdoc}
     */
    
protected function getBinaryTypeDeclarationSQLSnippet($length$fixed)
    {
        return 
'RAW(' . ($length ?: $this->getBinaryMaxLength()) . ')';
    }

    
/**
     * {@inheritdoc}
     */
    
public function getBinaryMaxLength()
    {
        return 
2000;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getClobTypeDeclarationSQL(array $column)
    {
        return 
'CLOB';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListDatabasesSQL()
    {
        return 
'SELECT username FROM all_users';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListSequencesSQL($database)
    {
        
$database $this->normalizeIdentifier($database);
        
$database $this->quoteStringLiteral($database->getName());

        return 
'SELECT sequence_name, min_value, increment_by FROM sys.all_sequences ' .
               
'WHERE SEQUENCE_OWNER = ' $database;
    }

    
/**
     * {@inheritDoc}
     */
    
protected function _getCreateTableSQL($name, array $columns, array $options = [])
    {
        
$indexes            $options['indexes'] ?? [];
        
$options['indexes'] = [];
        
$sql                parent::_getCreateTableSQL($name$columns$options);

        foreach (
$columns as $columnName => $column) {
            if (isset(
$column['sequence'])) {
                
$sql[] = $this->getCreateSequenceSQL($column['sequence']);
            }

            if (
                ! isset(
$column['autoincrement']) || ! $column['autoincrement'] &&
                (! isset(
$column['autoinc']) || ! $column['autoinc'])
            ) {
                continue;
            }

            
$sql array_merge($sql$this->getCreateAutoincrementSql($columnName$name));
        }

        if (isset(
$indexes) && ! empty($indexes)) {
            foreach (
$indexes as $index) {
                
$sql[] = $this->getCreateIndexSQL($index$name);
            }
        }

        return 
$sql;
    }

    
/**
     * {@inheritDoc}
     *
     * @link http://ezcomponents.org/docs/api/trunk/DatabaseSchema/ezcDbSchemaOracleReader.html
     */
    
public function getListTableIndexesSQL($table$database null)
    {
        
$table $this->normalizeIdentifier($table);
        
$table $this->quoteStringLiteral($table->getName());

        return 
"SELECT uind_col.index_name AS name,
                       (
                           SELECT uind.index_type
                           FROM   user_indexes uind
                           WHERE  uind.index_name = uind_col.index_name
                       ) AS type,
                       decode(
                           (
                               SELECT uind.uniqueness
                               FROM   user_indexes uind
                               WHERE  uind.index_name = uind_col.index_name
                           ),
                           'NONUNIQUE',
                           0,
                           'UNIQUE',
                           1
                       ) AS is_unique,
                       uind_col.column_name AS column_name,
                       uind_col.column_position AS column_pos,
                       (
                           SELECT ucon.constraint_type
                           FROM   user_constraints ucon
                           WHERE  ucon.index_name = uind_col.index_name
                       ) AS is_primary
             FROM      user_ind_columns uind_col
             WHERE     uind_col.table_name = " 
$table '
             ORDER BY  uind_col.column_position ASC'
;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListTablesSQL()
    {
        return 
'SELECT * FROM sys.user_tables';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListViewsSQL($database)
    {
        return 
'SELECT view_name, text FROM sys.user_views';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getCreateViewSQL($name$sql)
    {
        return 
'CREATE VIEW ' $name ' AS ' $sql;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDropViewSQL($name)
    {
        return 
'DROP VIEW ' $name;
    }

    
/**
     * @param string $name
     * @param string $table
     * @param int    $start
     *
     * @return string[]
     */
    
public function getCreateAutoincrementSql($name$table$start 1)
    {
        
$tableIdentifier   $this->normalizeIdentifier($table);
        
$quotedTableName   $tableIdentifier->getQuotedName($this);
        
$unquotedTableName $tableIdentifier->getName();

        
$nameIdentifier $this->normalizeIdentifier($name);
        
$quotedName     $nameIdentifier->getQuotedName($this);
        
$unquotedName   $nameIdentifier->getName();

        
$sql = [];

        
$autoincrementIdentifierName $this->getAutoincrementIdentifierName($tableIdentifier);

        
$idx = new Index($autoincrementIdentifierName, [$quotedName], truetrue);

        
$sql[] = "DECLARE
  constraints_Count NUMBER;
BEGIN
  SELECT COUNT(CONSTRAINT_NAME) INTO constraints_Count
    FROM USER_CONSTRAINTS
   WHERE TABLE_NAME = '" 
$unquotedTableName "'
     AND CONSTRAINT_TYPE = 'P';
  IF constraints_Count = 0 OR constraints_Count = '' THEN
    EXECUTE IMMEDIATE '" 
$this->getCreateConstraintSQL($idx$quotedTableName) . "';
  END IF;
END;"
;

        
$sequenceName $this->getIdentitySequenceName(
            
$tableIdentifier->isQuoted() ? $quotedTableName $unquotedTableName,
            
$nameIdentifier->isQuoted() ? $quotedName $unquotedName
        
);
        
$sequence     = new Sequence($sequenceName$start);
        
$sql[]        = $this->getCreateSequenceSQL($sequence);

        
$sql[] = 'CREATE TRIGGER ' $autoincrementIdentifierName '
   BEFORE INSERT
   ON ' 
$quotedTableName '
   FOR EACH ROW
DECLARE
   last_Sequence NUMBER;
   last_InsertID NUMBER;
BEGIN
   SELECT ' 
$sequenceName '.NEXTVAL INTO :NEW.' $quotedName ' FROM DUAL;
   IF (:NEW.' 
$quotedName ' IS NULL OR :NEW.' $quotedName ' = 0) THEN
      SELECT ' 
$sequenceName '.NEXTVAL INTO :NEW.' $quotedName ' FROM DUAL;
   ELSE
      SELECT NVL(Last_Number, 0) INTO last_Sequence
        FROM User_Sequences
       WHERE Sequence_Name = '' . $sequence->getName() . '';
      SELECT :NEW.' 
$quotedName ' INTO last_InsertID FROM DUAL;
      WHILE (last_InsertID > last_Sequence) LOOP
         SELECT ' 
$sequenceName '.NEXTVAL INTO last_Sequence FROM DUAL;
      END LOOP;
   END IF;
END;'
;

        return 
$sql;
    }

    
/**
     * Returns the SQL statements to drop the autoincrement for the given table name.
     *
     * @param string $table The table name to drop the autoincrement for.
     *
     * @return string[]
     */
    
public function getDropAutoincrementSql($table)
    {
        
$table                       $this->normalizeIdentifier($table);
        
$autoincrementIdentifierName $this->getAutoincrementIdentifierName($table);
        
$identitySequenceName        $this->getIdentitySequenceName(
            
$table->isQuoted() ? $table->getQuotedName($this) : $table->getName(),
            
''
        
);

        return [
            
'DROP TRIGGER ' $autoincrementIdentifierName,
            
$this->getDropSequenceSQL($identitySequenceName),
            
$this->getDropConstraintSQL($autoincrementIdentifierName$table->getQuotedName($this)),
        ];
    }

    
/**
     * Normalizes the given identifier.
     *
     * Uppercases the given identifier if it is not quoted by intention
     * to reflect Oracle's internal auto uppercasing strategy of unquoted identifiers.
     *
     * @param string $name The identifier to normalize.
     *
     * @return Identifier The normalized identifier.
     */
    
private function normalizeIdentifier($name)
    {
        
$identifier = new Identifier($name);

        return 
$identifier->isQuoted() ? $identifier : new Identifier(strtoupper($name));
    }

    
/**
     * Adds suffix to identifier,
     *
     * if the new string exceeds max identifier length,
     * keeps $suffix, cuts from $identifier as much as the part exceeding.
     */
    
private function addSuffix(string $identifierstring $suffix): string
    
{
        
$maxPossibleLengthWithoutSuffix $this->getMaxIdentifierLength() - strlen($suffix);
        if (
strlen($identifier) > $maxPossibleLengthWithoutSuffix) {
            
$identifier substr($identifier0$maxPossibleLengthWithoutSuffix);
        }

        return 
$identifier $suffix;
    }

    
/**
     * Returns the autoincrement primary key identifier name for the given table identifier.
     *
     * Quotes the autoincrement primary key identifier name
     * if the given table name is quoted by intention.
     *
     * @param Identifier $table The table identifier to return the autoincrement primary key identifier name for.
     *
     * @return string
     */
    
private function getAutoincrementIdentifierName(Identifier $table)
    {
        
$identifierName $this->addSuffix($table->getName(), '_AI_PK');

        return 
$table->isQuoted()
            ? 
$this->quoteSingleIdentifier($identifierName)
            : 
$identifierName;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListTableForeignKeysSQL($table)
    {
        
$table $this->normalizeIdentifier($table);
        
$table $this->quoteStringLiteral($table->getName());

        return 
"SELECT alc.constraint_name,
          alc.DELETE_RULE,
          cols.column_name "
local_column",
          cols.position,
          (
              SELECT r_cols.table_name
              FROM   user_cons_columns r_cols
              WHERE  alc.r_constraint_name = r_cols.constraint_name
              AND    r_cols.position = cols.position
          ) AS "
references_table",
          (
              SELECT r_cols.column_name
              FROM   user_cons_columns r_cols
              WHERE  alc.r_constraint_name = r_cols.constraint_name
              AND    r_cols.position = cols.position
          ) AS "
foreign_column"
     FROM user_cons_columns cols
     JOIN user_constraints alc
       ON alc.constraint_name = cols.constraint_name
      AND alc.constraint_type = 'R'
      AND alc.table_name = " 
$table '
    ORDER BY cols.constraint_name ASC, cols.position ASC'
;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListTableConstraintsSQL($table)
    {
        
$table $this->normalizeIdentifier($table);
        
$table $this->quoteStringLiteral($table->getName());

        return 
'SELECT * FROM user_constraints WHERE table_name = ' $table;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getListTableColumnsSQL($table$database null)
    {
        
$table $this->normalizeIdentifier($table);
        
$table $this->quoteStringLiteral($table->getName());

        
$tabColumnsTableName       'user_tab_columns';
        
$colCommentsTableName      'user_col_comments';
        
$tabColumnsOwnerCondition  '';
        
$colCommentsOwnerCondition '';

        if (
$database !== null && $database !== '/') {
            
$database                  $this->normalizeIdentifier($database);
            
$database                  $this->quoteStringLiteral($database->getName());
            
$tabColumnsTableName       'all_tab_columns';
            
$colCommentsTableName      'all_col_comments';
            
$tabColumnsOwnerCondition  ' AND c.owner = ' $database;
            
$colCommentsOwnerCondition ' AND d.OWNER = c.OWNER';
        }

        return 
sprintf(
            <<<'SQL'
SELECT   c.*,
         (
             SELECT d.comments
             FROM   %s d
             WHERE  d.TABLE_NAME = c.TABLE_NAME%s
             AND    d.COLUMN_NAME = c.COLUMN_NAME
         ) AS comments
FROM     %s c
WHERE    c.table_name = %s%s
ORDER BY c.column_id
SQL
            ,
            
$colCommentsTableName,
            
$colCommentsOwnerCondition,
            
$tabColumnsTableName,
            
$table,
            
$tabColumnsOwnerCondition
        
);
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDropSequenceSQL($sequence)
    {
        if (
$sequence instanceof Sequence) {
            
$sequence $sequence->getQuotedName($this);
        }

        return 
'DROP SEQUENCE ' $sequence;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDropForeignKeySQL($foreignKey$table)
    {
        if (! 
$foreignKey instanceof ForeignKeyConstraint) {
            
$foreignKey = new Identifier($foreignKey);
        }

        if (! 
$table instanceof Table) {
            
$table = new Identifier($table);
        }

        
$foreignKey $foreignKey->getQuotedName($this);
        
$table      $table->getQuotedName($this);

        return 
'ALTER TABLE ' $table ' DROP CONSTRAINT ' $foreignKey;
    }

    
/**
     * {@inheritdoc}
     */
    
public function getAdvancedForeignKeyOptionsSQL(ForeignKeyConstraint $foreignKey)
    {
        
$referentialAction null;

        if (
$foreignKey->hasOption('onDelete')) {
            
$referentialAction $this->getForeignKeyReferentialActionSQL($foreignKey->getOption('onDelete'));
        }

        return 
$referentialAction ' ON DELETE ' $referentialAction '';
    }

    
/**
     * {@inheritdoc}
     */
    
public function getForeignKeyReferentialActionSQL($action)
    {
        
$action strtoupper($action);

        switch (
$action) {
            case 
'RESTRICT'// RESTRICT is not supported, therefore falling back to NO ACTION.
            
case 'NO ACTION':
                
// NO ACTION cannot be declared explicitly,
                // therefore returning empty string to indicate to OMIT the referential clause.
                
return '';

            case 
'CASCADE':
            case 
'SET NULL':
                return 
$action;

            default:
                
// SET DEFAULT is not supported, throw exception instead.
                
throw new InvalidArgumentException('Invalid foreign key action: ' $action);
        }
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDropDatabaseSQL($name)
    {
        return 
'DROP USER ' $name ' CASCADE';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getAlterTableSQL(TableDiff $diff)
    {
        
$sql         = [];
        
$commentsSQL = [];
        
$columnSql   = [];

        
$fields = [];

        foreach (
$diff->addedColumns as $column) {
            if (
$this->onSchemaAlterTableAddColumn($column$diff$columnSql)) {
                continue;
            }

            
$fields[] = $this->getColumnDeclarationSQL($column->getQuotedName($this), $column->toArray());
            
$comment  $this->getColumnComment($column);

            if (! 
$comment) {
                continue;
            }

            
$commentsSQL[] = $this->getCommentOnColumnSQL(
                
$diff->getName($this)->getQuotedName($this),
                
$column->getQuotedName($this),
                
$comment
            
);
        }

        if (
count($fields)) {
            
$sql[] = 'ALTER TABLE ' $diff->getName($this)->getQuotedName($this)
                . 
' ADD (' implode(', '$fields) . ')';
        }

        
$fields = [];
        foreach (
$diff->changedColumns as $columnDiff) {
            if (
$this->onSchemaAlterTableChangeColumn($columnDiff$diff$columnSql)) {
                continue;
            }

            
$column $columnDiff->column;

            
// Do not generate column alteration clause if type is binary and only fixed property has changed.
            // Oracle only supports binary type columns with variable length.
            // Avoids unnecessary table alteration statements.
            
if (
                
$column->getType() instanceof BinaryType &&
                
$columnDiff->hasChanged('fixed') &&
                
count($columnDiff->changedProperties) === 1
            
) {
                continue;
            }

            
$columnHasChangedComment $columnDiff->hasChanged('comment');

            
/**
             * Do not add query part if only comment has changed
             */
            
if (! ($columnHasChangedComment && count($columnDiff->changedProperties) === 1)) {
                
$columnInfo $column->toArray();

                if (! 
$columnDiff->hasChanged('notnull')) {
                    unset(
$columnInfo['notnull']);
                }

                
$fields[] = $column->getQuotedName($this) . $this->getColumnDeclarationSQL(''$columnInfo);
            }

            if (! 
$columnHasChangedComment) {
                continue;
            }

            
$commentsSQL[] = $this->getCommentOnColumnSQL(
                
$diff->getName($this)->getQuotedName($this),
                
$column->getQuotedName($this),
                
$this->getColumnComment($column)
            );
        }

        if (
count($fields)) {
            
$sql[] = 'ALTER TABLE ' $diff->getName($this)->getQuotedName($this)
                . 
' MODIFY (' implode(', '$fields) . ')';
        }

        foreach (
$diff->renamedColumns as $oldColumnName => $column) {
            if (
$this->onSchemaAlterTableRenameColumn($oldColumnName$column$diff$columnSql)) {
                continue;
            }

            
$oldColumnName = new Identifier($oldColumnName);

            
$sql[] = 'ALTER TABLE ' $diff->getName($this)->getQuotedName($this) .
                
' RENAME COLUMN ' $oldColumnName->getQuotedName($this) . ' TO ' $column->getQuotedName($this);
        }

        
$fields = [];
        foreach (
$diff->removedColumns as $column) {
            if (
$this->onSchemaAlterTableRemoveColumn($column$diff$columnSql)) {
                continue;
            }

            
$fields[] = $column->getQuotedName($this);
        }

        if (
count($fields)) {
            
$sql[] = 'ALTER TABLE ' $diff->getName($this)->getQuotedName($this)
                . 
' DROP (' implode(', '$fields) . ')';
        }

        
$tableSql = [];

        if (! 
$this->onSchemaAlterTable($diff$tableSql)) {
            
$sql array_merge($sql$commentsSQL);

            
$newName $diff->getNewName();

            if (
$newName !== false) {
                
$sql[] = sprintf(
                    
'ALTER TABLE %s RENAME TO %s',
                    
$diff->getName($this)->getQuotedName($this),
                    
$newName->getQuotedName($this)
                );
            }

            
$sql array_merge(
                
$this->getPreAlterTableIndexForeignKeySQL($diff),
                
$sql,
                
$this->getPostAlterTableIndexForeignKeySQL($diff)
            );
        }

        return 
array_merge($sql$tableSql$columnSql);
    }

    
/**
     * {@inheritdoc}
     */
    
public function getColumnDeclarationSQL($name, array $column)
    {
        if (isset(
$column['columnDefinition'])) {
            
$columnDef $this->getCustomTypeDeclarationSQL($column);
        } else {
            
$default $this->getDefaultValueDeclarationSQL($column);

            
$notnull '';

            if (isset(
$column['notnull'])) {
                
$notnull $column['notnull'] ? ' NOT NULL' ' NULL';
            }

            
$unique = isset($column['unique']) && $column['unique'] ?
                
' ' $this->getUniqueFieldDeclarationSQL() : '';

            
$check = isset($column['check']) && $column['check'] ?
                
' ' $column['check'] : '';

            
$typeDecl  $column['type']->getSQLDeclaration($column$this);
            
$columnDef $typeDecl $default $notnull $unique $check;
        }

        return 
$name ' ' $columnDef;
    }

    
/**
     * {@inheritdoc}
     */
    
protected function getRenameIndexSQL($oldIndexNameIndex $index$tableName)
    {
        if (
strpos($tableName'.') !== false) {
            [
$schema]     = explode('.'$tableName);
            
$oldIndexName $schema '.' $oldIndexName;
        }

        return [
'ALTER INDEX ' $oldIndexName ' RENAME TO ' $index->getQuotedName($this)];
    }

    
/**
     * {@inheritDoc}
     *
     * @deprecated
     */
    
public function prefersSequences()
    {
        
Deprecation::trigger(
            
'doctrine/dbal',
            
'https://github.com/doctrine/dbal/pull/4229',
            
'AbstractPlatform::prefersSequences() is deprecated without replacement and removed in DBAL 3.0'
        
);

        return 
true;
    }

    
/**
     * {@inheritdoc}
     */
    
public function usesSequenceEmulatedIdentityColumns()
    {
        return 
true;
    }

    
/**
     * {@inheritdoc}
     */
    
public function getIdentitySequenceName($tableName$columnName)
    {
        
$table = new Identifier($tableName);

        
// No usage of column name to preserve BC compatibility with <2.5
        
$identitySequenceName $this->addSuffix($table->getName(), '_SEQ');

        if (
$table->isQuoted()) {
            
$identitySequenceName '"' $identitySequenceName '"';
        }

        
$identitySequenceIdentifier $this->normalizeIdentifier($identitySequenceName);

        return 
$identitySequenceIdentifier->getQuotedName($this);
    }

    
/**
     * {@inheritDoc}
     */
    
public function supportsCommentOnStatement()
    {
        return 
true;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getName()
    {
        return 
'oracle';
    }

    
/**
     * {@inheritDoc}
     */
    
protected function doModifyLimitQuery($query$limit$offset null)
    {
        if (
$limit === null && $offset <= 0) {
            return 
$query;
        }

        if (
preg_match('/^s*SELECT/i'$query)) {
            if (! 
preg_match('/sFROMs/i'$query)) {
                
$query .= ' FROM dual';
            }

            
$columns = ['a.*'];

            if (
$offset 0) {
                
$columns[] = 'ROWNUM AS doctrine_rownum';
            }

            
$query sprintf('SELECT %s FROM (%s) a'implode(', '$columns), $query);

            if (
$limit !== null) {
                
$query .= sprintf(' WHERE ROWNUM <= %d'$offset $limit);
            }

            if (
$offset 0) {
                
$query sprintf('SELECT * FROM (%s) WHERE doctrine_rownum >= %d'$query$offset 1);
            }
        }

        return 
$query;
    }

    
/**
     * {@inheritDoc}
     *
     * Oracle returns all column names in SQL result sets in uppercase.
     *
     * @deprecated
     */
    
public function getSQLResultCasing($column)
    {
        return 
strtoupper($column);
    }

    
/**
     * {@inheritDoc}
     */
    
public function getCreateTemporaryTableSnippetSQL()
    {
        return 
'CREATE GLOBAL TEMPORARY TABLE';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateTimeTzFormatString()
    {
        return 
'Y-m-d H:i:sP';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDateFormatString()
    {
        return 
'Y-m-d 00:00:00';
    }

    
/**
     * {@inheritDoc}
     */
    
public function getTimeFormatString()
    {
        return 
'1900-01-01 H:i:s';
    }

    
/**
     * {@inheritDoc}
     *
     * @deprecated
     */
    
public function fixSchemaElementName($schemaElementName)
    {
        
Deprecation::trigger(
            
'doctrine/dbal',
            
'https://github.com/doctrine/dbal/pull/4132',
            
'AbstractPlatform::fixSchemaElementName is deprecated with no replacement and removed in DBAL 3.0'
        
);

        if (
strlen($schemaElementName) > 30) {
            
// Trim it
            
return substr($schemaElementName030);
        }

        return 
$schemaElementName;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getMaxIdentifierLength()
    {
        return 
30;
    }

    
/**
     * {@inheritDoc}
     */
    
public function supportsSequences()
    {
        return 
true;
    }

    
/**
     * {@inheritDoc}
     *
     * @deprecated
     */
    
public function supportsForeignKeyOnUpdate()
    {
        
Deprecation::trigger(
            
'doctrine/dbal',
            
'https://github.com/doctrine/dbal/pull/4229',
            
'AbstractPlatform::supportsForeignKeyOnUpdate() is deprecated without replacement and removed in DBAL 3.0'
        
);

        return 
false;
    }

    
/**
     * {@inheritDoc}
     */
    
public function supportsReleaseSavepoints()
    {
        return 
false;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getTruncateTableSQL($tableName$cascade false)
    {
        
$tableIdentifier = new Identifier($tableName);

        return 
'TRUNCATE TABLE ' $tableIdentifier->getQuotedName($this);
    }

    
/**
     * {@inheritDoc}
     */
    
public function getDummySelectSQL()
    {
        
$expression func_num_args() > func_get_arg(0) : '1';

        return 
sprintf('SELECT %s FROM DUAL'$expression);
    }

    
/**
     * {@inheritDoc}
     */
    
protected function initializeDoctrineTypeMappings()
    {
        
$this->doctrineTypeMapping = [
            
'integer'           => 'integer',
            
'number'            => 'integer',
            
'pls_integer'       => 'boolean',
            
'binary_integer'    => 'boolean',
            
'varchar'           => 'string',
            
'varchar2'          => 'string',
            
'nvarchar2'         => 'string',
            
'char'              => 'string',
            
'nchar'             => 'string',
            
'date'              => 'date',
            
'timestamp'         => 'datetime',
            
'timestamptz'       => 'datetimetz',
            
'float'             => 'float',
            
'binary_float'      => 'float',
            
'binary_double'     => 'float',
            
'long'              => 'string',
            
'clob'              => 'text',
            
'nclob'             => 'text',
            
'raw'               => 'binary',
            
'long raw'          => 'blob',
            
'rowid'             => 'string',
            
'urowid'            => 'string',
            
'blob'              => 'blob',
        ];
    }

    
/**
     * {@inheritDoc}
     */
    
public function releaseSavePoint($savepoint)
    {
        return 
'';
    }

    
/**
     * {@inheritDoc}
     */
    
protected function getReservedKeywordsClass()
    {
        return 
KeywordsOracleKeywords::class;
    }

    
/**
     * {@inheritDoc}
     */
    
public function getBlobTypeDeclarationSQL(array $column)
    {
        return 
'BLOB';
    }

    public function 
getListTableCommentsSQL(string $table, ?string $database null): string
    
{
        
$tableCommentsName 'user_tab_comments';
        
$ownerCondition    '';

        if (
$database !== null && $database !== '/') {
            
$tableCommentsName 'all_tab_comments';
            
$ownerCondition    ' AND owner = ' $this->quoteStringLiteral(
                
$this->normalizeIdentifier($database)->getName()
            );
        }

        return 
sprintf(
            <<<'SQL'
SELECT comments FROM %s WHERE table_name = %s%s
SQL
            ,
            
$tableCommentsName,
            
$this->quoteStringLiteral($this->normalizeIdentifier($table)->getName()),
            
$ownerCondition
        
);
    }
}
Онлайн: 0
Реклама