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.
286 lines
7.5 KiB
286 lines
7.5 KiB
1 year ago
|
<?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\Commands\Database;
|
||
|
|
||
|
use CodeIgniter\CLI\BaseCommand;
|
||
|
use CodeIgniter\CLI\CLI;
|
||
|
use CodeIgniter\Database\BaseConnection;
|
||
|
use Config\Database;
|
||
|
|
||
|
/**
|
||
|
* Get table data if it exists in the database.
|
||
|
*
|
||
|
* @see \CodeIgniter\Commands\Database\ShowTableInfoTest
|
||
|
*/
|
||
|
class ShowTableInfo extends BaseCommand
|
||
|
{
|
||
|
/**
|
||
|
* The group the command is lumped under
|
||
|
* when listing commands.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $group = 'Database';
|
||
|
|
||
|
/**
|
||
|
* The Command's name
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $name = 'db:table';
|
||
|
|
||
|
/**
|
||
|
* the Command's short description
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $description = 'Retrieves information on the selected table.';
|
||
|
|
||
|
/**
|
||
|
* the Command's usage
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $usage = <<<'EOL'
|
||
|
db:table [<table_name>] [options]
|
||
|
|
||
|
Examples:
|
||
|
db:table --show
|
||
|
db:table --metadata
|
||
|
db:table my_table --metadata
|
||
|
db:table my_table
|
||
|
db:table my_table --limit-rows 5 --limit-field-value 10 --desc
|
||
|
EOL;
|
||
|
|
||
|
/**
|
||
|
* The Command's arguments
|
||
|
*
|
||
|
* @var array<string, string>
|
||
|
*/
|
||
|
protected $arguments = [
|
||
|
'table_name' => 'The table name to show info',
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* The Command's options
|
||
|
*
|
||
|
* @var array<string, string>
|
||
|
*/
|
||
|
protected $options = [
|
||
|
'--show' => 'Lists the names of all database tables.',
|
||
|
'--metadata' => 'Retrieves list containing field information.',
|
||
|
'--desc' => 'Sorts the table rows in DESC order.',
|
||
|
'--limit-rows' => 'Limits the number of rows. Default: 10.',
|
||
|
'--limit-field-value' => 'Limits the length of field values. Default: 15.',
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* @var list<list<int|string>> Table Data.
|
||
|
*/
|
||
|
private array $tbody;
|
||
|
|
||
|
private BaseConnection $db;
|
||
|
|
||
|
/**
|
||
|
* @var bool Sort the table rows in DESC order or not.
|
||
|
*/
|
||
|
private bool $sortDesc = false;
|
||
|
|
||
|
private string $DBPrefix;
|
||
|
|
||
|
public function run(array $params)
|
||
|
{
|
||
|
$this->db = Database::connect();
|
||
|
$this->DBPrefix = $this->db->getPrefix();
|
||
|
|
||
|
$tables = $this->db->listTables();
|
||
|
|
||
|
if (array_key_exists('desc', $params)) {
|
||
|
$this->sortDesc = true;
|
||
|
}
|
||
|
|
||
|
if ($tables === []) {
|
||
|
CLI::error('Database has no tables!', 'light_gray', 'red');
|
||
|
CLI::newLine();
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('show', $params)) {
|
||
|
$this->showAllTables($tables);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$tableName = $params[0] ?? null;
|
||
|
$limitRows = (int) ($params['limit-rows'] ?? 10);
|
||
|
$limitFieldValue = (int) ($params['limit-field-value'] ?? 15);
|
||
|
|
||
|
while (! in_array($tableName, $tables, true)) {
|
||
|
$tableNameNo = CLI::promptByKey(
|
||
|
['Here is the list of your database tables:', 'Which table do you want to see?'],
|
||
|
$tables,
|
||
|
'required'
|
||
|
);
|
||
|
CLI::newLine();
|
||
|
|
||
|
$tableName = $tables[$tableNameNo] ?? null;
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('metadata', $params)) {
|
||
|
$this->showFieldMetaData($tableName);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$this->showDataOfTable($tableName, $limitRows, $limitFieldValue);
|
||
|
}
|
||
|
|
||
|
private function removeDBPrefix(): void
|
||
|
{
|
||
|
$this->db->setPrefix('');
|
||
|
}
|
||
|
|
||
|
private function restoreDBPrefix(): void
|
||
|
{
|
||
|
$this->db->setPrefix($this->DBPrefix);
|
||
|
}
|
||
|
|
||
|
private function showDataOfTable(string $tableName, int $limitRows, int $limitFieldValue)
|
||
|
{
|
||
|
CLI::write("Data of Table \"{$tableName}\":", 'black', 'yellow');
|
||
|
CLI::newLine();
|
||
|
|
||
|
$this->removeDBPrefix();
|
||
|
$thead = $this->db->getFieldNames($tableName);
|
||
|
$this->restoreDBPrefix();
|
||
|
|
||
|
// If there is a field named `id`, sort by it.
|
||
|
$sortField = null;
|
||
|
if (in_array('id', $thead, true)) {
|
||
|
$sortField = 'id';
|
||
|
}
|
||
|
|
||
|
$this->tbody = $this->makeTableRows($tableName, $limitRows, $limitFieldValue, $sortField);
|
||
|
CLI::table($this->tbody, $thead);
|
||
|
}
|
||
|
|
||
|
private function showAllTables(array $tables)
|
||
|
{
|
||
|
CLI::write('The following is a list of the names of all database tables:', 'black', 'yellow');
|
||
|
CLI::newLine();
|
||
|
|
||
|
$thead = ['ID', 'Table Name', 'Num of Rows', 'Num of Fields'];
|
||
|
$this->tbody = $this->makeTbodyForShowAllTables($tables);
|
||
|
|
||
|
CLI::table($this->tbody, $thead);
|
||
|
CLI::newLine();
|
||
|
}
|
||
|
|
||
|
private function makeTbodyForShowAllTables(array $tables): array
|
||
|
{
|
||
|
$this->removeDBPrefix();
|
||
|
|
||
|
foreach ($tables as $id => $tableName) {
|
||
|
$table = $this->db->protectIdentifiers($tableName);
|
||
|
$db = $this->db->query("SELECT * FROM {$table}");
|
||
|
|
||
|
$this->tbody[] = [
|
||
|
$id + 1,
|
||
|
$tableName,
|
||
|
$db->getNumRows(),
|
||
|
$db->getFieldCount(),
|
||
|
];
|
||
|
}
|
||
|
|
||
|
$this->restoreDBPrefix();
|
||
|
|
||
|
if ($this->sortDesc) {
|
||
|
krsort($this->tbody);
|
||
|
}
|
||
|
|
||
|
return $this->tbody;
|
||
|
}
|
||
|
|
||
|
private function makeTableRows(
|
||
|
string $tableName,
|
||
|
int $limitRows,
|
||
|
int $limitFieldValue,
|
||
|
?string $sortField = null
|
||
|
): array {
|
||
|
$this->tbody = [];
|
||
|
|
||
|
$this->removeDBPrefix();
|
||
|
$builder = $this->db->table($tableName);
|
||
|
$builder->limit($limitRows);
|
||
|
if ($sortField !== null) {
|
||
|
$builder->orderBy($sortField, $this->sortDesc ? 'DESC' : 'ASC');
|
||
|
}
|
||
|
$rows = $builder->get()->getResultArray();
|
||
|
$this->restoreDBPrefix();
|
||
|
|
||
|
foreach ($rows as $row) {
|
||
|
$row = array_map(
|
||
|
static fn ($item): string => mb_strlen((string) $item) > $limitFieldValue
|
||
|
? mb_substr((string) $item, 0, $limitFieldValue) . '...'
|
||
|
: (string) $item,
|
||
|
$row
|
||
|
);
|
||
|
$this->tbody[] = $row;
|
||
|
}
|
||
|
|
||
|
if ($sortField === null && $this->sortDesc) {
|
||
|
krsort($this->tbody);
|
||
|
}
|
||
|
|
||
|
return $this->tbody;
|
||
|
}
|
||
|
|
||
|
private function showFieldMetaData(string $tableName): void
|
||
|
{
|
||
|
CLI::write("List of Metadata Information in Table \"{$tableName}\":", 'black', 'yellow');
|
||
|
CLI::newLine();
|
||
|
|
||
|
$thead = ['Field Name', 'Type', 'Max Length', 'Nullable', 'Default', 'Primary Key'];
|
||
|
|
||
|
$this->removeDBPrefix();
|
||
|
$fields = $this->db->getFieldData($tableName);
|
||
|
$this->restoreDBPrefix();
|
||
|
|
||
|
foreach ($fields as $row) {
|
||
|
$this->tbody[] = [
|
||
|
$row->name,
|
||
|
$row->type,
|
||
|
$row->max_length,
|
||
|
isset($row->nullable) ? $this->setYesOrNo($row->nullable) : 'n/a',
|
||
|
$row->default,
|
||
|
isset($row->primary_key) ? $this->setYesOrNo($row->primary_key) : 'n/a',
|
||
|
];
|
||
|
}
|
||
|
|
||
|
if ($this->sortDesc) {
|
||
|
krsort($this->tbody);
|
||
|
}
|
||
|
|
||
|
CLI::table($this->tbody, $thead);
|
||
|
}
|
||
|
|
||
|
private function setYesOrNo(bool $fieldValue): string
|
||
|
{
|
||
|
if ($fieldValue) {
|
||
|
return CLI::color('Yes', 'green');
|
||
|
}
|
||
|
|
||
|
return CLI::color('No', 'red');
|
||
|
}
|
||
|
}
|