You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
488 lines
12 KiB
488 lines
12 KiB
<?php |
|
|
|
/** |
|
* This file is part of CodeIgniter 4 framework. |
|
* |
|
* (c) CodeIgniter Foundation <admin@codeigniter.com> |
|
* |
|
* For the full copyright and license information, please view |
|
* the LICENSE file that was distributed with this source code. |
|
*/ |
|
|
|
namespace CodeIgniter\Database\SQLite3; |
|
|
|
use CodeIgniter\Database\Exceptions\DataException; |
|
use stdClass; |
|
|
|
/** |
|
* Class Table |
|
* |
|
* Provides missing features for altering tables that are common |
|
* in other supported databases, but are missing from SQLite. |
|
* These are needed in order to support migrations during testing |
|
* when another database is used as the primary engine, but |
|
* SQLite in memory databases are used for faster test execution. |
|
*/ |
|
class Table |
|
{ |
|
/** |
|
* All of the fields this table represents. |
|
* |
|
* @var array<string, array<string, bool|int|string|null>> [name => attributes] |
|
*/ |
|
protected $fields = []; |
|
|
|
/** |
|
* All of the unique/primary keys in the table. |
|
* |
|
* @var array |
|
*/ |
|
protected $keys = []; |
|
|
|
/** |
|
* All of the foreign keys in the table. |
|
* |
|
* @var array |
|
*/ |
|
protected $foreignKeys = []; |
|
|
|
/** |
|
* The name of the table we're working with. |
|
* |
|
* @var string |
|
*/ |
|
protected $tableName; |
|
|
|
/** |
|
* The name of the table, with database prefix |
|
* |
|
* @var string |
|
*/ |
|
protected $prefixedTableName; |
|
|
|
/** |
|
* Database connection. |
|
* |
|
* @var Connection |
|
*/ |
|
protected $db; |
|
|
|
/** |
|
* Handle to our forge. |
|
* |
|
* @var Forge |
|
*/ |
|
protected $forge; |
|
|
|
/** |
|
* Table constructor. |
|
*/ |
|
public function __construct(Connection $db, Forge $forge) |
|
{ |
|
$this->db = $db; |
|
$this->forge = $forge; |
|
} |
|
|
|
/** |
|
* Reads an existing database table and |
|
* collects all of the information needed to |
|
* recreate this table. |
|
* |
|
* @return Table |
|
*/ |
|
public function fromTable(string $table) |
|
{ |
|
$this->prefixedTableName = $table; |
|
|
|
$prefix = $this->db->DBPrefix; |
|
|
|
if (! empty($prefix) && strpos($table, $prefix) === 0) { |
|
$table = substr($table, strlen($prefix)); |
|
} |
|
|
|
if (! $this->db->tableExists($this->prefixedTableName)) { |
|
throw DataException::forTableNotFound($this->prefixedTableName); |
|
} |
|
|
|
$this->tableName = $table; |
|
|
|
$this->fields = $this->formatFields($this->db->getFieldData($table)); |
|
|
|
$this->keys = array_merge($this->keys, $this->formatKeys($this->db->getIndexData($table))); |
|
|
|
// if primary key index exists twice then remove psuedo index name 'primary'. |
|
$primaryIndexes = array_filter($this->keys, static fn ($index) => $index['type'] === 'primary'); |
|
|
|
if ($primaryIndexes !== [] && count($primaryIndexes) > 1 && array_key_exists('primary', $this->keys)) { |
|
unset($this->keys['primary']); |
|
} |
|
|
|
$this->foreignKeys = $this->db->getForeignKeyData($table); |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Called after `fromTable` and any actions, like `dropColumn`, etc, |
|
* to finalize the action. It creates a temp table, creates the new |
|
* table with modifications, and copies the data over to the new table. |
|
* Resets the connection dataCache to be sure changes are collected. |
|
*/ |
|
public function run(): bool |
|
{ |
|
$this->db->query('PRAGMA foreign_keys = OFF'); |
|
|
|
$this->db->transStart(); |
|
|
|
$this->forge->renameTable($this->tableName, "temp_{$this->tableName}"); |
|
|
|
$this->forge->reset(); |
|
|
|
$this->createTable(); |
|
|
|
$this->copyData(); |
|
|
|
$this->forge->dropTable("temp_{$this->tableName}"); |
|
|
|
$success = $this->db->transComplete(); |
|
|
|
$this->db->query('PRAGMA foreign_keys = ON'); |
|
|
|
$this->db->resetDataCache(); |
|
|
|
return $success; |
|
} |
|
|
|
/** |
|
* Drops columns from the table. |
|
* |
|
* @param list<string>|string $columns Column names to drop. |
|
* |
|
* @return Table |
|
*/ |
|
public function dropColumn($columns) |
|
{ |
|
if (is_string($columns)) { |
|
$columns = explode(',', $columns); |
|
} |
|
|
|
foreach ($columns as $column) { |
|
$column = trim($column); |
|
if (isset($this->fields[$column])) { |
|
unset($this->fields[$column]); |
|
} |
|
} |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Modifies a field, including changing data type, renaming, etc. |
|
* |
|
* @param list<array<string, bool|int|string|null>> $fieldsToModify |
|
* |
|
* @return Table |
|
*/ |
|
public function modifyColumn(array $fieldsToModify) |
|
{ |
|
foreach ($fieldsToModify as $field) { |
|
$oldName = $field['name']; |
|
unset($field['name']); |
|
|
|
$this->fields[$oldName] = $field; |
|
} |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Drops the primary key |
|
*/ |
|
public function dropPrimaryKey(): Table |
|
{ |
|
$primaryIndexes = array_filter($this->keys, static fn ($index) => strtolower($index['type']) === 'primary'); |
|
|
|
foreach (array_keys($primaryIndexes) as $key) { |
|
unset($this->keys[$key]); |
|
} |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Drops a foreign key from this table so that |
|
* it won't be recreated in the future. |
|
* |
|
* @return Table |
|
*/ |
|
public function dropForeignKey(string $foreignName) |
|
{ |
|
if (empty($this->foreignKeys)) { |
|
return $this; |
|
} |
|
|
|
if (isset($this->foreignKeys[$foreignName])) { |
|
unset($this->foreignKeys[$foreignName]); |
|
} |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Adds primary key |
|
*/ |
|
public function addPrimaryKey(array $fields): Table |
|
{ |
|
$primaryIndexes = array_filter($this->keys, static fn ($index) => strtolower($index['type']) === 'primary'); |
|
|
|
// if primary key already exists we can't add another one |
|
if ($primaryIndexes !== []) { |
|
return $this; |
|
} |
|
|
|
// add array to keys of fields |
|
$pk = [ |
|
'fields' => $fields['fields'], |
|
'type' => 'primary', |
|
]; |
|
|
|
$this->keys['primary'] = $pk; |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Add a foreign key |
|
* |
|
* @return $this |
|
*/ |
|
public function addForeignKey(array $foreignKeys) |
|
{ |
|
$fk = []; |
|
|
|
// convert to object |
|
foreach ($foreignKeys as $row) { |
|
$obj = new stdClass(); |
|
$obj->column_name = $row['field']; |
|
$obj->foreign_table_name = $row['referenceTable']; |
|
$obj->foreign_column_name = $row['referenceField']; |
|
$obj->on_delete = $row['onDelete']; |
|
$obj->on_update = $row['onUpdate']; |
|
|
|
$fk[] = $obj; |
|
} |
|
|
|
$this->foreignKeys = array_merge($this->foreignKeys, $fk); |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Creates the new table based on our current fields. |
|
* |
|
* @return bool |
|
*/ |
|
protected function createTable() |
|
{ |
|
$this->dropIndexes(); |
|
$this->db->resetDataCache(); |
|
|
|
// Handle any modified columns. |
|
$fields = []; |
|
|
|
foreach ($this->fields as $name => $field) { |
|
if (isset($field['new_name'])) { |
|
$fields[$field['new_name']] = $field; |
|
|
|
continue; |
|
} |
|
|
|
$fields[$name] = $field; |
|
} |
|
|
|
$this->forge->addField($fields); |
|
|
|
$fieldNames = array_keys($fields); |
|
|
|
$this->keys = array_filter( |
|
$this->keys, |
|
static fn ($index) => count(array_intersect($index['fields'], $fieldNames)) === count($index['fields']) |
|
); |
|
|
|
// Unique/Index keys |
|
if (is_array($this->keys)) { |
|
foreach ($this->keys as $keyName => $key) { |
|
switch ($key['type']) { |
|
case 'primary': |
|
$this->forge->addPrimaryKey($key['fields']); |
|
break; |
|
|
|
case 'unique': |
|
$this->forge->addUniqueKey($key['fields'], $keyName); |
|
break; |
|
|
|
case 'index': |
|
$this->forge->addKey($key['fields'], false, false, $keyName); |
|
break; |
|
} |
|
} |
|
} |
|
|
|
foreach ($this->foreignKeys as $foreignKey) { |
|
$this->forge->addForeignKey( |
|
$foreignKey->column_name, |
|
trim($foreignKey->foreign_table_name, $this->db->DBPrefix), |
|
$foreignKey->foreign_column_name |
|
); |
|
} |
|
|
|
return $this->forge->createTable($this->tableName); |
|
} |
|
|
|
/** |
|
* Copies data from our old table to the new one, |
|
* taking care map data correctly based on any columns |
|
* that have been renamed. |
|
*/ |
|
protected function copyData() |
|
{ |
|
$exFields = []; |
|
$newFields = []; |
|
|
|
foreach ($this->fields as $name => $details) { |
|
$newFields[] = $details['new_name'] ?? $name; |
|
$exFields[] = $name; |
|
} |
|
|
|
$exFields = implode( |
|
', ', |
|
array_map(fn ($item) => $this->db->protectIdentifiers($item), $exFields) |
|
); |
|
$newFields = implode( |
|
', ', |
|
array_map(fn ($item) => $this->db->protectIdentifiers($item), $newFields) |
|
); |
|
|
|
$this->db->query( |
|
"INSERT INTO {$this->prefixedTableName}({$newFields}) SELECT {$exFields} FROM {$this->db->DBPrefix}temp_{$this->tableName}" |
|
); |
|
} |
|
|
|
/** |
|
* Converts fields retrieved from the database to |
|
* the format needed for creating fields with Forge. |
|
* |
|
* @param array|bool $fields |
|
* |
|
* @return mixed |
|
* @phpstan-return ($fields is array ? array : mixed) |
|
*/ |
|
protected function formatFields($fields) |
|
{ |
|
if (! is_array($fields)) { |
|
return $fields; |
|
} |
|
|
|
$return = []; |
|
|
|
foreach ($fields as $field) { |
|
$return[$field->name] = [ |
|
'type' => $field->type, |
|
'default' => $field->default, |
|
'null' => $field->nullable, |
|
]; |
|
|
|
if ($field->default === null) { |
|
// `null` means that the default value is not defined. |
|
unset($return[$field->name]['default']); |
|
} elseif ($field->default === 'NULL') { |
|
// 'NULL' means that the default value is NULL. |
|
$return[$field->name]['default'] = null; |
|
} else { |
|
$default = trim($field->default, "'"); |
|
|
|
if ($this->isIntegerType($field->type)) { |
|
$default = (int) $default; |
|
} elseif ($this->isNumericType($field->type)) { |
|
$default = (float) $default; |
|
} |
|
|
|
$return[$field->name]['default'] = $default; |
|
} |
|
|
|
if ($field->primary_key) { |
|
$this->keys['primary'] = [ |
|
'fields' => [$field->name], |
|
'type' => 'primary', |
|
]; |
|
} |
|
} |
|
|
|
return $return; |
|
} |
|
|
|
/** |
|
* Is INTEGER type? |
|
* |
|
* @param string $type SQLite data type (case-insensitive) |
|
* |
|
* @see https://www.sqlite.org/datatype3.html |
|
*/ |
|
private function isIntegerType(string $type): bool |
|
{ |
|
return strpos(strtoupper($type), 'INT') !== false; |
|
} |
|
|
|
/** |
|
* Is NUMERIC type? |
|
* |
|
* @param string $type SQLite data type (case-insensitive) |
|
* |
|
* @see https://www.sqlite.org/datatype3.html |
|
*/ |
|
private function isNumericType(string $type): bool |
|
{ |
|
return in_array(strtoupper($type), ['NUMERIC', 'DECIMAL'], true); |
|
} |
|
|
|
/** |
|
* Converts keys retrieved from the database to |
|
* the format needed to create later. |
|
* |
|
* @param array<string, stdClass> $keys |
|
* |
|
* @return array<string, array{fields: string, type: string}> |
|
*/ |
|
protected function formatKeys($keys) |
|
{ |
|
$return = []; |
|
|
|
foreach ($keys as $name => $key) { |
|
$return[strtolower($name)] = [ |
|
'fields' => $key->fields, |
|
'type' => strtolower($key->type), |
|
]; |
|
} |
|
|
|
return $return; |
|
} |
|
|
|
/** |
|
* Attempts to drop all indexes and constraints |
|
* from the database for this table. |
|
*/ |
|
protected function dropIndexes() |
|
{ |
|
if (! is_array($this->keys) || $this->keys === []) { |
|
return; |
|
} |
|
|
|
foreach (array_keys($this->keys) as $name) { |
|
if ($name === 'primary') { |
|
continue; |
|
} |
|
|
|
$this->db->query("DROP INDEX IF EXISTS '{$name}'"); |
|
} |
|
} |
|
}
|
|
|