src/Database/Cycle/Generator/ShowChanges.php
<?php declare(strict_types=1);
/*
* This file is part of Biurad opensource projects.
*
* @copyright 2019 Biurad Group (https://biurad.com/)
* @license https://opensource.org/licenses/BSD-3-Clause License
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Flange\Database\Cycle\Generator;
use Cycle\Database\Schema\AbstractTable;
use Cycle\Database\Schema\Comparator;
use Cycle\Schema\GeneratorInterface;
use Cycle\Schema\Registry;
use Symfony\Component\Console\Output\OutputInterface;
final class ShowChanges implements GeneratorInterface
{
private OutputInterface $output;
/** @var array<string,array<string,string|AbstractTable>> */
private array $changes = [];
public function __construct(OutputInterface $output)
{
$this->output = $output;
}
public function run(Registry $registry): Registry
{
$this->output->writeln('<info>Detecting schema changes:</info>'."\n");
$this->changes = [];
foreach ($registry->getIterator() as $e) {
if ($registry->hasTable($e)) {
$table = $registry->getTableSchema($e);
if ($table->getComparator()->hasChanges()) {
$key = $registry->getDatabase($e).':'.$registry->getTable($e);
$this->changes[$key] = [
'database' => $registry->getDatabase($e),
'table' => $registry->getTable($e),
'schema' => $table,
];
}
}
}
if ([] === $this->changes) {
$this->output->writeln('<fg=yellow>No database changes has been detected</fg=yellow>');
return $registry;
}
foreach ($this->changes as $change) {
$this->output->write(\sprintf('• <fg=cyan>%s.%s</fg=cyan>', $change['database'], $change['table']));
$this->describeChanges($change['schema']);
}
return $registry;
}
public function hasChanges(): bool
{
return [] !== $this->changes;
}
protected function describeChanges(AbstractTable $table): void
{
if (!$this->output->isVerbose()) {
$this->output->writeln(
\sprintf(
': <fg=green>%s</fg=green> change(s) detected',
$this->numChanges($table)
)
);
return;
}
$this->output->write("\n");
if (!$table->exists()) {
$this->output->writeln(' - create table');
}
if (AbstractTable::STATUS_DECLARED_DROPPED === $table->getStatus()) {
$this->output->writeln(' - drop table');
return;
}
$cmp = $table->getComparator();
$this->describeColumns($cmp);
$this->describeIndexes($cmp);
$this->describeFKs($cmp);
}
protected function describeColumns(Comparator $cmp): void
{
foreach ($cmp->addedColumns() as $column) {
$this->output->writeln(" - add column <fg=yellow>{$column->getName()}</fg=yellow>");
}
foreach ($cmp->droppedColumns() as $column) {
$this->output->writeln(" - drop column <fg=yellow>{$column->getName()}</fg=yellow>");
}
foreach ($cmp->alteredColumns() as $column) {
$column = $column[0];
$this->output->writeln(" - alter column <fg=yellow>{$column->getName()}</fg=yellow>");
}
}
protected function describeIndexes(Comparator $cmp): void
{
foreach ($cmp->addedIndexes() as $index) {
$index = \implode(', ', $index->getColumns());
$this->output->writeln(" - add index on <fg=yellow>[{$index}]</fg=yellow>");
}
foreach ($cmp->droppedIndexes() as $index) {
$index = \implode(', ', $index->getColumns());
$this->output->writeln(" - drop index on <fg=yellow>[{$index}]</fg=yellow>");
}
foreach ($cmp->alteredIndexes() as $index) {
$index = $index[0];
$index = \implode(', ', $index->getColumns());
$this->output->writeln(" - alter index on <fg=yellow>[{$index}]</fg=yellow>");
}
}
protected function describeFKs(Comparator $cmp): void
{
foreach ($cmp->addedForeignKeys() as $fk) {
$fkColumns = \implode(', ', $fk->getColumns());
$this->output->writeln(" - add foreign key on <fg=yellow>{$fkColumns}</fg=yellow>");
}
foreach ($cmp->droppedForeignKeys() as $fk) {
$fkColumns = \implode(', ', $fk->getColumns());
$this->output->writeln(" - drop foreign key <fg=yellow>{$fkColumns}</fg=yellow>");
}
foreach ($cmp->alteredForeignKeys() as $fk) {
$fk = $fk[0];
$fkColumns = \implode(', ', $fk->getColumns());
$this->output->writeln(" - alter foreign key <fg=yellow>{$fkColumns}</fg=yellow>");
}
}
protected function numChanges(AbstractTable $table): int
{
$cmp = $table->getComparator();
return \count($cmp->addedColumns())
+ \count($cmp->droppedColumns())
+ \count($cmp->alteredColumns())
+ \count($cmp->addedIndexes())
+ \count($cmp->droppedIndexes())
+ \count($cmp->alteredIndexes())
+ \count($cmp->addedForeignKeys())
+ \count($cmp->droppedForeignKeys())
+ \count($cmp->alteredForeignKeys());
}
}