dbi = $dbi; $this->tableObj = $this->dbi->getTable($this->db, $this->table); } public function __invoke(): void { if (! isset($_POST['move_columns']) || ! is_array($_POST['move_columns']) || ! $this->response->isAjax()) { return; } $this->dbi->selectDb($this->db); /** * load the definitions for all columns */ $columns = $this->dbi->getColumnsFull($this->db, $this->table); $column_names = array_keys($columns); $changes = []; // @see https://mariadb.com/kb/en/library/changes-improvements-in-mariadb-102/#information-schema $usesLiteralNull = $this->dbi->isMariaDB() && $this->dbi->getVersion() >= 100200; $defaultNullValue = $usesLiteralNull ? 'NULL' : null; // move columns from first to last for ($i = 0, $l = count($_POST['move_columns']); $i < $l; $i++) { $column = $_POST['move_columns'][$i]; // is this column already correctly placed? if ($column_names[$i] == $column) { continue; } // it is not, let's move it to index $i $data = $columns[$column]; $extracted_columnspec = Util::extractColumnSpec($data['Type']); if (isset($data['Extra']) && $data['Extra'] === 'on update CURRENT_TIMESTAMP') { $extracted_columnspec['attribute'] = $data['Extra']; unset($data['Extra']); } $timeType = $data['Type'] === 'timestamp' || $data['Type'] === 'datetime'; $timeDefault = $data['Default'] === 'CURRENT_TIMESTAMP' || $data['Default'] === 'current_timestamp()'; $current_timestamp = $timeType && $timeDefault; // @see https://mariadb.com/kb/en/library/information-schema-columns-table/#examples if ($data['Null'] === 'YES' && in_array($data['Default'], [$defaultNullValue, null])) { $default_type = 'NULL'; } elseif ($current_timestamp) { $default_type = 'CURRENT_TIMESTAMP'; } elseif ($data['Default'] === null) { $default_type = 'NONE'; } else { $default_type = 'USER_DEFINED'; } $virtual = [ 'VIRTUAL', 'PERSISTENT', 'VIRTUAL GENERATED', 'STORED GENERATED', ]; $data['Virtuality'] = ''; $data['Expression'] = ''; if (isset($data['Extra']) && in_array($data['Extra'], $virtual)) { $data['Virtuality'] = str_replace(' GENERATED', '', $data['Extra']); $expressions = $this->tableObj->getColumnGenerationExpression($column); $data['Expression'] = is_array($expressions) ? $expressions[$column] : null; } $changes[] = 'CHANGE ' . Table::generateAlter( $column, $column, mb_strtoupper($extracted_columnspec['type']), $extracted_columnspec['spec_in_brackets'], $extracted_columnspec['attribute'], $data['Collation'] ?? '', $data['Null'] === 'YES' ? 'YES' : 'NO', $default_type, $current_timestamp ? '' : $data['Default'], isset($data['Extra']) && $data['Extra'] !== '' ? $data['Extra'] : false, isset($data['COLUMN_COMMENT']) && $data['COLUMN_COMMENT'] !== '' ? $data['COLUMN_COMMENT'] : false, $data['Virtuality'], $data['Expression'], $i === 0 ? '-first' : $column_names[$i - 1] ); // update current column_names array, first delete old position for ($j = 0, $ll = count($column_names); $j < $ll; $j++) { if ($column_names[$j] != $column) { continue; } unset($column_names[$j]); } // insert moved column array_splice($column_names, $i, 0, $column); } if (empty($changes) && ! isset($_REQUEST['preview_sql'])) { // should never happen $this->response->setRequestStatus(false); return; } // query for moving the columns $sql_query = sprintf( 'ALTER TABLE %s %s', Util::backquote($this->table), implode(', ', $changes) ); if (isset($_REQUEST['preview_sql'])) { // preview sql $this->response->addJSON( 'sql_data', $this->template->render('preview_sql', ['query_data' => $sql_query]) ); return; } $this->dbi->tryQuery($sql_query); $tmp_error = $this->dbi->getError(); if ($tmp_error !== '') { $this->response->setRequestStatus(false); $this->response->addJSON('message', Message::error($tmp_error)); return; } $message = Message::success( __('The columns have been moved successfully.') ); $this->response->addJSON('message', $message); $this->response->addJSON('columns', $column_names); } }