Example #1
0
 /**
  * Gets the SQL to alter an existing table.
  *
  * @param string $name The name of the table that is intended to be changed.
  * @param array $changes Associative array that contains the details of each type
  *                             of change that is intended to be performed. The types of
  *                             changes that are currently supported are defined as follows:
  *
  *                             name
  *
  *                                New name for the table.
  *
  *                            add
  *
  *                                Associative array with the names of fields to be added as
  *                                 indexes of the array. The value of each entry of the array
  *                                 should be set to another associative array with the properties
  *                                 of the fields to be added. The properties of the fields should
  *                                 be the same as defined by the Metabase parser.
  *
  *
  *                            remove
  *
  *                                Associative array with the names of fields to be removed as indexes
  *                                 of the array. Currently the values assigned to each entry are ignored.
  *                                 An empty array should be used for future compatibility.
  *
  *                            rename
  *
  *                                Associative array with the names of fields to be renamed as indexes
  *                                 of the array. The value of each entry of the array should be set to
  *                                 another associative array with the entry named name with the new
  *                                 field name and the entry named Declaration that is expected to contain
  *                                 the portion of the field declaration already in DBMS specific SQL code
  *                                 as it is used in the CREATE TABLE statement.
  *
  *                            change
  *
  *                                Associative array with the names of the fields to be changed as indexes
  *                                 of the array. Keep in mind that if it is intended to change either the
  *                                 name of a field and any other properties, the change array entries
  *                                 should have the new names of the fields as array indexes.
  *
  *                                The value of each entry of the array should be set to another associative
  *                                 array with the properties of the fields to that are meant to be changed as
  *                                 array entries. These entries should be assigned to the new values of the
  *                                 respective properties. The properties of the fields should be the same
  *                                 as defined by the Metabase parser.
  *
  *                            Example
  *                                array(
  *                                    'name' => 'userlist',
  *                                    'add' => array(
  *                                        'quota' => array(
  *                                            'type' => 'integer',
  *                                            'unsigned' => 1
  *                                        )
  *                                    ),
  *                                    'remove' => array(
  *                                        'file_limit' => array(),
  *                                        'time_limit' => array()
  *                                    ),
  *                                    'change' => array(
  *                                        'name' => array(
  *                                            'length' => '20',
  *                                            'definition' => array(
  *                                                'type' => 'text',
  *                                                'length' => 20,
  *                                            ),
  *                                        )
  *                                    ),
  *                                    'rename' => array(
  *                                        'sex' => array(
  *                                            'name' => 'gender',
  *                                            'definition' => array(
  *                                                'type' => 'text',
  *                                                'length' => 1,
  *                                                'default' => 'M',
  *                                            ),
  *                                        )
  *                                    )
  *                                )
  *
  * @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.
  * @return boolean
  * @override
  */
 public function getAlterTableSql($name, array $changes, $check = false)
 {
     if (!$name) {
         throw DoctrineException::missingTableName();
     }
     foreach ($changes as $changeName => $change) {
         switch ($changeName) {
             case 'add':
             case 'remove':
             case 'change':
             case 'rename':
             case 'name':
                 break;
             default:
                 throw DoctrineException::alterTableChangeNotSupported($changeName);
         }
     }
     if ($check) {
         return true;
     }
     $query = '';
     if (!empty($changes['name'])) {
         $query .= 'RENAME TO ' . $changes['name'];
     }
     if (!empty($changes['add']) && is_array($changes['add'])) {
         foreach ($changes['add'] as $fieldName => $field) {
             if ($query) {
                 $query .= ', ';
             }
             $query .= 'ADD ' . $this->getColumnDeclarationSql($fieldName, $field);
         }
     }
     if (!empty($changes['remove']) && is_array($changes['remove'])) {
         foreach ($changes['remove'] as $fieldName => $field) {
             if ($query) {
                 $query .= ', ';
             }
             $query .= 'DROP ' . $fieldName;
         }
     }
     $rename = array();
     if (!empty($changes['rename']) && is_array($changes['rename'])) {
         foreach ($changes['rename'] as $fieldName => $field) {
             $rename[$field['name']] = $fieldName;
         }
     }
     if (!empty($changes['change']) && is_array($changes['change'])) {
         foreach ($changes['change'] as $fieldName => $field) {
             if ($query) {
                 $query .= ', ';
             }
             if (isset($rename[$fieldName])) {
                 $oldFieldName = $rename[$fieldName];
                 unset($rename[$fieldName]);
             } else {
                 $oldFieldName = $fieldName;
             }
             $query .= 'CHANGE ' . $oldFieldName . ' ' . $this->getColumnDeclarationSql($fieldName, $field['definition']);
         }
     }
     if (!empty($rename) && is_array($rename)) {
         foreach ($rename as $renameName => $renamedField) {
             if ($query) {
                 $query .= ', ';
             }
             $field = $changes['rename'][$renamedField];
             $query .= 'CHANGE ' . $renamedField . ' ' . $this->getColumnDeclarationSql($field['name'], $field['definition']);
         }
     }
     if (!$query) {
         return false;
     }
     return 'ALTER TABLE ' . $name . ' ' . $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:
                 throw DoctrineException::alterTableChangeNotSupported($changeName);
         }
     }
     if ($check) {
         return true;
     }
     $sql = array();
     if (isset($changes['add']) && is_array($changes['add'])) {
         foreach ($changes['add'] as $fieldName => $field) {
             $query = 'ADD ' . $this->getColumnDeclarationSql($fieldName, $field);
             $sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
         }
     }
     if (isset($changes['remove']) && is_array($changes['remove'])) {
         foreach ($changes['remove'] as $fieldName => $field) {
             $query = 'DROP ' . $fieldName;
             $sql[] = 'ALTER TABLE ' . $name . ' ' . $query;
         }
     }
     if (isset($changes['change']) && is_array($changes['change'])) {
         foreach ($changes['change'] as $fieldName => $field) {
             if (isset($field['type'])) {
                 $serverInfo = $this->getServerVersion();
                 if (is_array($serverInfo) && $serverInfo['major'] < 8) {
                     throw DoctrineException::changeColumnRequiresPostgreSQL8OrAbove($field['type']);
                 }
                 $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) {
             $sql[] = 'ALTER TABLE ' . $name . ' RENAME COLUMN ' . $fieldName . ' TO ' . $field['name'];
         }
     }
     if (isset($changes['name'])) {
         $sql[] = 'ALTER TABLE ' . $name . ' RENAME TO ' . $changes['name'];
     }
     return $sql;
 }