diff --git a/src/Console/Command/CycleOrm/Generator/ShowChanges.php b/src/Console/Command/CycleOrm/Generator/ShowChanges.php
index f65292b..7591983 100644
--- a/src/Console/Command/CycleOrm/Generator/ShowChanges.php
+++ b/src/Console/Command/CycleOrm/Generator/ShowChanges.php
@@ -4,10 +4,10 @@
namespace Spiral\Cycle\Console\Command\CycleOrm\Generator;
+use Cycle\Database\Schema\ComparatorInterface;
use Cycle\Schema\GeneratorInterface;
use Cycle\Schema\Registry;
use Cycle\Database\Schema\AbstractTable;
-use Cycle\Database\Schema\Comparator;
use Symfony\Component\Console\Output\OutputInterface;
final class ShowChanges implements GeneratorInterface
@@ -15,20 +15,18 @@ final class ShowChanges implements GeneratorInterface
private array $changes = [];
public function __construct(
- private readonly OutputInterface $output
+ private readonly OutputInterface $output,
) {
}
public function run(Registry $registry): Registry
{
- $this->output->writeln('Detecting schema changes:');
-
$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);
+ $key = $registry->getDatabase($e) . ':' . $registry->getTable($e);
$this->changes[$key] = [
'database' => $registry->getDatabase($e),
'table' => $registry->getTable($e),
@@ -44,6 +42,9 @@ public function run(Registry $registry): Registry
return $registry;
}
+
+ $this->output->writeln('Schema changes:');
+
foreach ($this->changes as $change) {
$this->output->write(\sprintf('• %s.%s', $change['database'], $change['table']));
$this->describeChanges($change['schema']);
@@ -63,14 +64,14 @@ protected function describeChanges(AbstractTable $table): void
$this->output->writeln(
\sprintf(
': %s change(s) detected',
- $this->numChanges($table)
- )
+ $this->numChanges($table),
+ ),
);
return;
}
- $this->output->write("\n");
+ $this->output->write("\n");
if (!$table->exists()) {
$this->output->writeln(' - create table');
@@ -88,7 +89,7 @@ protected function describeChanges(AbstractTable $table): void
$this->describeFKs($cmp);
}
- protected function describeColumns(Comparator $cmp): void
+ protected function describeColumns(ComparatorInterface $cmp): void
{
foreach ($cmp->addedColumns() as $column) {
$this->output->writeln(" - add column {$column->getName()}");
@@ -104,7 +105,7 @@ protected function describeColumns(Comparator $cmp): void
}
}
- protected function describeIndexes(Comparator $cmp): void
+ protected function describeIndexes(ComparatorInterface $cmp): void
{
foreach ($cmp->addedIndexes() as $index) {
$index = \implode(', ', $index->getColumns());
@@ -123,7 +124,7 @@ protected function describeIndexes(Comparator $cmp): void
}
}
- protected function describeFKs(Comparator $cmp): void
+ protected function describeFKs(ComparatorInterface $cmp): void
{
foreach ($cmp->addedForeignKeys() as $fk) {
$fkColumns = \implode(', ', $fk->getColumns());
diff --git a/src/Console/Command/CycleOrm/MigrateCommand.php b/src/Console/Command/CycleOrm/MigrateCommand.php
index 3ce1873..86dc90d 100644
--- a/src/Console/Command/CycleOrm/MigrateCommand.php
+++ b/src/Console/Command/CycleOrm/MigrateCommand.php
@@ -4,6 +4,8 @@
namespace Spiral\Cycle\Console\Command\CycleOrm;
+use Cycle\Schema\Generator\Migrations\Strategy\GeneratorStrategyInterface;
+use Cycle\Schema\Generator\Migrations\Strategy\MultipleFilesStrategy;
use Spiral\Cycle\Bootloader\SchemaBootloader;
use Spiral\Cycle\Config\CycleConfig;
use Spiral\Cycle\Console\Command\CycleOrm\Generator\ShowChanges;
@@ -22,6 +24,7 @@ final class MigrateCommand extends AbstractCommand
protected const NAME = 'cycle:migrate';
protected const DESCRIPTION = 'Generate ORM schema migrations';
protected const OPTIONS = [
+ ['split', 'p', InputOption::VALUE_NONE, 'Split generated migration into multiple files.'],
['run', 'r', InputOption::VALUE_NONE, 'Automatically run generated migration.'],
];
@@ -30,30 +33,43 @@ public function perform(
CycleConfig $config,
Registry $registry,
MemoryInterface $memory,
- GenerateMigrations $migrations,
Migrator $migrator,
- Console $console
+ Console $console,
): int {
$migrator->configure();
foreach ($migrator->getMigrations() as $migration) {
if ($migration->getState()->getStatus() !== State::STATUS_EXECUTED) {
- $this->writeln('Outstanding migrations found, run `migrate` first.');
- return self::SUCCESS;
+ $this->error('Outstanding migrations found.');
+
+ if ($this->isInteractive() && $this->output->confirm('Do you want to run `migrate` now?')) {
+ $console->run('migrate', [], $this->output);
+ } else {
+ $this->error('Please run `migrate` first.');
+ return self::SUCCESS;
+ }
}
}
+ $this->comment('Detecting schema changes...');
+
$schemaCompiler = Compiler::compile(
$registry,
\array_merge($bootloader->getGenerators($config), [
- $show = new ShowChanges($this->output)
+ $show = new ShowChanges($this->output),
]),
- $config->getSchemaDefaults()
+ $config->getSchemaDefaults(),
);
$schemaCompiler->toMemory($memory);
if ($show->hasChanges()) {
+ if ($this->option('split')) {
+ $this->container->bind(GeneratorStrategyInterface::class, MultipleFilesStrategy::class);
+ }
+
+ $migrations = $this->container->get(GenerateMigrations::class);
+
(new \Cycle\Schema\Compiler())->compile($registry, [$migrations]);
if ($this->option('run')) {
diff --git a/src/Console/Command/CycleOrm/RenderCommand.php b/src/Console/Command/CycleOrm/RenderCommand.php
index a6527f3..5b0ecf3 100644
--- a/src/Console/Command/CycleOrm/RenderCommand.php
+++ b/src/Console/Command/CycleOrm/RenderCommand.php
@@ -15,7 +15,6 @@
final class RenderCommand extends AbstractCommand
{
protected const SIGNATURE = 'cycle:render {format=color : Output format}';
-
protected const DESCRIPTION = 'Render available CycleORM schemas';
public function perform(
diff --git a/src/Console/Command/CycleOrm/SyncCommand.php b/src/Console/Command/CycleOrm/SyncCommand.php
index 680cb89..12d0352 100644
--- a/src/Console/Command/CycleOrm/SyncCommand.php
+++ b/src/Console/Command/CycleOrm/SyncCommand.php
@@ -8,12 +8,12 @@
use Cycle\Schema\Generator\SyncTables;
use Cycle\Schema\Registry;
use Spiral\Boot\MemoryInterface;
-use Spiral\Console\Command;
use Spiral\Cycle\Config\CycleConfig;
use Spiral\Cycle\Console\Command\CycleOrm\Generator\ShowChanges;
+use Spiral\Cycle\Console\Command\Migrate\AbstractCommand;
use Spiral\Cycle\Schema\Compiler;
-final class SyncCommand extends Command
+final class SyncCommand extends AbstractCommand
{
protected const NAME = 'cycle:sync';
protected const DESCRIPTION = 'Sync Cycle ORM schema with database without intermediate migration (risk operation)';
@@ -22,19 +22,24 @@ public function perform(
SchemaBootloader $bootloader,
CycleConfig $config,
Registry $registry,
- MemoryInterface $memory
+ MemoryInterface $memory,
): int {
+ if (!$this->verifyEnvironment(message: 'This operation is not recommended for production environment.')) {
+ return self::FAILURE;
+ }
+
$show = new ShowChanges($this->output);
$schemaCompiler = Compiler::compile(
$registry,
\array_merge($bootloader->getGenerators($config), [$show, new SyncTables()]),
- $config->getSchemaDefaults()
+ $config->getSchemaDefaults(),
);
+
$schemaCompiler->toMemory($memory);
if ($show->hasChanges()) {
- $this->writeln("\nORM Schema has been synchronized");
+ $this->info('ORM Schema has been synchronized with database.');
}
return self::SUCCESS;
diff --git a/src/Console/Command/CycleOrm/UpdateCommand.php b/src/Console/Command/CycleOrm/UpdateCommand.php
index d5eb8c7..ef83522 100644
--- a/src/Console/Command/CycleOrm/UpdateCommand.php
+++ b/src/Console/Command/CycleOrm/UpdateCommand.php
@@ -20,17 +20,17 @@ public function perform(
SchemaBootloader $bootloader,
CycleConfig $config,
Registry $registry,
- MemoryInterface $memory
+ MemoryInterface $memory,
): int {
- $this->write('Updating ORM schema... ');
+ $this->info('Updating ORM schema... ');
Compiler::compile(
$registry,
$bootloader->getGenerators($config),
- $config->getSchemaDefaults()
+ $config->getSchemaDefaults(),
)->toMemory($memory);
- $this->writeln('done');
+ $this->info('Schema has been updated.');
return self::SUCCESS;
}
diff --git a/src/Console/Command/Migrate/AbstractCommand.php b/src/Console/Command/Migrate/AbstractCommand.php
index 271f8b0..853305d 100644
--- a/src/Console/Command/Migrate/AbstractCommand.php
+++ b/src/Console/Command/Migrate/AbstractCommand.php
@@ -15,7 +15,7 @@ abstract class AbstractCommand extends Command
{
public function __construct(
protected Migrator $migrator,
- protected MigrationConfig $config
+ protected MigrationConfig $config,
) {
parent::__construct();
}
@@ -24,7 +24,7 @@ protected function verifyConfigured(): bool
{
if (!$this->migrator->isConfigured()) {
$this->writeln(
- "Migrations are not configured yet, run 'migrate:init' first."
+ "Migrations are not configured yet, run 'migrate:init' first.",
);
return false;
@@ -36,17 +36,17 @@ protected function verifyConfigured(): bool
/**
* Check if current environment is safe to run migration.
*/
- protected function verifyEnvironment(): bool
+ protected function verifyEnvironment(string $message = 'Confirmation is required to run migrations!'): bool
{
- if ($this->option('force') || $this->config->isSafe()) {
+ if ($this->isForce() || $this->config->isSafe()) {
//Safe to run
return true;
}
- $this->writeln('Confirmation is required to run migrations!');
+ $this->error($message);
if (!$this->askConfirmation()) {
- $this->writeln('Cancelling operation...');
+ $this->comment('Cancelling operation...');
return false;
}
@@ -60,7 +60,8 @@ protected function defineOptions(): array
static::OPTIONS,
[
['force', 's', InputOption::VALUE_NONE, 'Skip safe environment check'],
- ]
+ ['no-interaction', 'n', InputOption::VALUE_NONE, 'Do not ask any interactive question'],
+ ],
);
}
@@ -71,7 +72,17 @@ protected function askConfirmation(): bool
return $question->ask(
$this->input,
$this->output,
- new ConfirmationQuestion('Would you like to continue? ')
+ new ConfirmationQuestion('Would you like to continue? ', false),
);
}
+
+ protected function isInteractive(): bool
+ {
+ return !$this->option('no-interaction');
+ }
+
+ protected function isForce(): bool
+ {
+ return $this->option('force');
+ }
}
diff --git a/src/Console/Command/Migrate/InitCommand.php b/src/Console/Command/Migrate/InitCommand.php
index 6faefec..e8c418c 100644
--- a/src/Console/Command/Migrate/InitCommand.php
+++ b/src/Console/Command/Migrate/InitCommand.php
@@ -7,7 +7,7 @@
final class InitCommand extends AbstractCommand
{
protected const NAME = 'migrate:init';
- protected const DESCRIPTION = 'Init migrations component (create migrations table)';
+ protected const DESCRIPTION = 'Create migrations table if not exists.';
/**
* Perform command.
@@ -15,7 +15,7 @@ final class InitCommand extends AbstractCommand
public function perform(): int
{
$this->migrator->configure();
- $this->writeln('Migrations table were successfully created');
+ $this->info('Migration table was successfully created.');
return self::SUCCESS;
}
diff --git a/src/Console/Command/Migrate/MigrateCommand.php b/src/Console/Command/Migrate/MigrateCommand.php
index c5884b7..8a2eb78 100644
--- a/src/Console/Command/Migrate/MigrateCommand.php
+++ b/src/Console/Command/Migrate/MigrateCommand.php
@@ -9,7 +9,7 @@
final class MigrateCommand extends AbstractCommand
{
protected const NAME = 'migrate';
- protected const DESCRIPTION = 'Perform one or all outstanding migrations';
+ protected const DESCRIPTION = 'Execute one or multiple migrations.';
protected const OPTIONS = [
['one', 'o', InputOption::VALUE_NONE, 'Execute only one (first) migration'],
];
@@ -34,12 +34,12 @@ public function perform(): int
$this->sprintf(
"Migration %s was successfully executed.\n",
- $migration->getState()->getName()
+ $migration->getState()->getName(),
);
}
if (!$found) {
- $this->writeln('No outstanding migrations were found.');
+ $this->error('No outstanding migrations were found.');
}
return self::SUCCESS;
diff --git a/src/Console/Command/Migrate/ReplayCommand.php b/src/Console/Command/Migrate/ReplayCommand.php
index 21c5f3e..eedd760 100644
--- a/src/Console/Command/Migrate/ReplayCommand.php
+++ b/src/Console/Command/Migrate/ReplayCommand.php
@@ -16,7 +16,6 @@ final class ReplayCommand extends AbstractCommand
];
/**
- * @param Console $console
* @throws \Throwable
*/
public function perform(Console $console): int
@@ -35,12 +34,12 @@ public function perform(Console $console): int
$migrate['--one'] = true;
}
- $this->writeln('Rolling back executed migration(s)...');
+ $this->warning('Rolling back executed migration(s)...');
$console->run('migrate:rollback', $rollback, $this->output);
$this->writeln('');
- $this->writeln('Executing outstanding migration(s)...');
+ $this->info('Executing outstanding migration(s)...');
$console->run('migrate', $migrate, $this->output);
return self::SUCCESS;
diff --git a/src/Console/Command/Migrate/RollbackCommand.php b/src/Console/Command/Migrate/RollbackCommand.php
index 4d9e429..2760419 100644
--- a/src/Console/Command/Migrate/RollbackCommand.php
+++ b/src/Console/Command/Migrate/RollbackCommand.php
@@ -17,7 +17,6 @@ final class RollbackCommand extends AbstractCommand
public function perform(): int
{
if (!$this->verifyEnvironment()) {
- //Making sure we can safely migrate in this environment
return self::FAILURE;
}
@@ -30,12 +29,12 @@ public function perform(): int
$count--;
$this->sprintf(
"Migration %s was successfully rolled back.\n",
- $migration->getState()->getName()
+ $migration->getState()->getName(),
);
}
if (!$found) {
- $this->writeln('No executed migrations were found.');
+ $this->error('No executed migrations were found.');
}
return self::SUCCESS;
diff --git a/src/Console/Command/Migrate/StatusCommand.php b/src/Console/Command/Migrate/StatusCommand.php
index 8f76647..13a79e3 100644
--- a/src/Console/Command/Migrate/StatusCommand.php
+++ b/src/Console/Command/Migrate/StatusCommand.php
@@ -7,12 +7,12 @@
use Cycle\Migrations\State;
/**
- * Show all available migrations and their statuses
+ * Get list of all available migrations and their statuses.
*/
final class StatusCommand extends AbstractCommand
{
protected const NAME = 'migrate:status';
- protected const DESCRIPTION = 'Get list of all available migrations and their statuses';
+ protected const DESCRIPTION = 'Get list of all available migrations and their statuses.';
protected const PENDING = 'not executed yet';
public function perform(): int
@@ -35,8 +35,8 @@ public function perform(): int
$state->getTimeCreated()->format('Y-m-d H:i:s'),
$state->getStatus() == State::STATUS_PENDING
? self::PENDING
- : ''.$state->getTimeExecuted()->format('Y-m-d H:i:s').'',
- ]
+ : '' . $state->getTimeExecuted()->format('Y-m-d H:i:s') . '',
+ ],
);
}
diff --git a/src/Console/Command/Scaffolder/EntityCommand.php b/src/Console/Command/Scaffolder/EntityCommand.php
index 3a44abc..929497f 100644
--- a/src/Console/Command/Scaffolder/EntityCommand.php
+++ b/src/Console/Command/Scaffolder/EntityCommand.php
@@ -19,12 +19,12 @@
class EntityCommand extends AbstractCommand
{
- protected const NAME = 'create:entity';
+ protected const NAME = 'create:entity';
protected const DESCRIPTION = 'Create entity declaration';
- protected const ARGUMENTS = [
+ protected const ARGUMENTS = [
['name', InputArgument::REQUIRED, 'Entity name'],
];
- protected const OPTIONS = [
+ protected const OPTIONS = [
[
'role',
'r',
diff --git a/src/Console/Command/Scaffolder/MigrationCommand.php b/src/Console/Command/Scaffolder/MigrationCommand.php
index 32bf488..8252ae2 100644
--- a/src/Console/Command/Scaffolder/MigrationCommand.php
+++ b/src/Console/Command/Scaffolder/MigrationCommand.php
@@ -13,12 +13,12 @@
class MigrationCommand extends AbstractCommand
{
- protected const NAME = 'create:migration';
+ protected const NAME = 'create:migration';
protected const DESCRIPTION = 'Create migration declaration';
- protected const ARGUMENTS = [
+ protected const ARGUMENTS = [
['name', InputArgument::REQUIRED, 'Migration name'],
];
- protected const OPTIONS = [
+ protected const OPTIONS = [
[
'table',
't',
@@ -66,12 +66,12 @@ public function perform(Migrator $migrator): int
$filename = $migrator->getRepository()->registerMigration(
(string)$this->argument('name'),
$declaration->getClass()->getName(),
- (string) $declaration->getFile()
+ (string)$declaration->getFile(),
);
$this->writeln(
"Declaration of '{$declaration->getClass()->getName()}' "
- . "has been successfully written into '{$filename}'."
+ . "has been successfully written into '{$filename}'.",
);
return self::SUCCESS;
diff --git a/tests/src/Console/Command/CycleOrm/MigrateCommandTest.php b/tests/src/Console/Command/CycleOrm/MigrateCommandTest.php
index 7961e3b..45c737f 100644
--- a/tests/src/Console/Command/CycleOrm/MigrateCommandTest.php
+++ b/tests/src/Console/Command/CycleOrm/MigrateCommandTest.php
@@ -4,8 +4,6 @@
namespace Spiral\Tests\Console\Command\CycleOrm;
-use Cycle\Annotated\Annotation\Column;
-use Cycle\Annotated\Annotation\Entity;
use Cycle\ORM\SchemaInterface;
use Spiral\Boot\MemoryInterface;
use Spiral\Cycle\Config\CycleConfig;
@@ -38,8 +36,15 @@ protected function setUp(): void
public function testMigrate(): void
{
+ // Create migration
$this->assertConsoleCommandOutputContainsStrings('cycle:migrate', [], self::USER_MIGRATION);
- $this->assertConsoleCommandOutputContainsStrings('cycle:migrate', [], 'Outstanding migrations found');
+
+ $this->assertConsoleCommandOutputContainsStrings('cycle:migrate', [], [
+ 'Outstanding migrations found',
+ 'Detecting schema changes...',
+ 'Migration 0_default_create_roles_create_users_create_auth_tokens was successfully executed.',
+ 'no database changes has been detected'
+ ]);
}
public function testMigrateNoChanges(): void
diff --git a/tests/src/Console/Command/CycleOrm/SyncCommandTest.php b/tests/src/Console/Command/CycleOrm/SyncCommandTest.php
index aef66c6..dc674aa 100644
--- a/tests/src/Console/Command/CycleOrm/SyncCommandTest.php
+++ b/tests/src/Console/Command/CycleOrm/SyncCommandTest.php
@@ -8,15 +8,24 @@
use Spiral\App\Entities\User;
use Spiral\Boot\MemoryInterface;
use Spiral\Cycle\Config\CycleConfig;
+use Spiral\Testing\Attribute\Env;
use Spiral\Tests\ConsoleTest;
final class SyncCommandTest extends ConsoleTest
{
public const ENV = [
- 'SAFE_MIGRATIONS' => true,
'USE_MIGRATIONS' => true,
];
+ #[Env('SAFE_MIGRATIONS', 'false')]
+ public function testUnsafeSync(): void
+ {
+ $output = $this->runCommand('cycle:sync');
+ $this->assertStringContainsString('This operation is not recommended for production environment.', $output);
+ $this->assertStringContainsString('Cancelling operation...', $output);
+ }
+
+ #[Env('SAFE_MIGRATIONS', 'true')]
public function testSync(): void
{
$output = $this->runCommand('cycle:sync');
@@ -28,6 +37,7 @@ public function testSync(): void
$this->assertSame(1, $u->id);
}
+ #[Env('SAFE_MIGRATIONS', 'true')]
public function testSyncDebug(): void
{
$this->assertConsoleCommandOutputContainsStrings('cycle:sync', ['-vvv'], [
@@ -43,6 +53,7 @@ public function testSyncDebug(): void
$this->assertSame(1, $u->id);
}
+ #[Env('SAFE_MIGRATIONS', 'true')]
public function testSchemaDefaultsShouldBePassedToCompiler(): void
{
$config['schema']['defaults'][SchemaInterface::TYPECAST_HANDLER][] = 'foo';
diff --git a/tests/src/Console/Command/Migrate/MigrateCommandTest.php b/tests/src/Console/Command/Migrate/MigrateCommandTest.php
index 7a5eba8..df97c67 100644
--- a/tests/src/Console/Command/Migrate/MigrateCommandTest.php
+++ b/tests/src/Console/Command/Migrate/MigrateCommandTest.php
@@ -5,16 +5,46 @@
namespace Spiral\Tests\Console\Command\Migrate;
use Cycle\Database\DatabaseInterface;
+use Spiral\Testing\Attribute\Env;
use Spiral\Tests\ConsoleTest;
final class MigrateCommandTest extends ConsoleTest
{
public const ENV = [
- 'SAFE_MIGRATIONS' => true,
'USE_MIGRATIONS' => true,
];
+ #[Env('SAFE_MIGRATIONS', true)]
public function testMigrate(): void
+ {
+ $db = $this->initMigrations();
+ $this->runCommand('migrate');
+ $this->assertCount(4, $db->getTables());
+ }
+
+ #[Env('SAFE_MIGRATIONS', false)]
+ public function tesForceMigrate(): void
+ {
+ $db = $this->initMigrations();
+ $this->runCommand('migrate', ['--force' => true]);
+ $this->assertCount(4, $db->getTables());
+ }
+
+ #[Env('SAFE_MIGRATIONS', false)]
+ public function testUnsafeMigrate(): void
+ {
+ $db = $this->initMigrations();
+ $output = $this->runCommand('migrate');
+ $this->assertStringContainsString('Confirmation is required to run migrations!', $output);
+ $this->assertStringContainsString('Cancelling operation...', $output);
+ $this->assertCount(1, $db->getTables());
+ }
+
+ /**
+ * @return void
+ * @throws \Throwable
+ */
+ public function initMigrations(): DatabaseInterface
{
/** @var DatabaseInterface $db */
$db = $this->getContainer()->get(DatabaseInterface::class);
@@ -25,7 +55,6 @@ public function testMigrate(): void
$this->assertCount(1, $db->getTables());
- $this->runCommand('migrate');
- $this->assertCount(4, $db->getTables());
+ return $db;
}
}