202 lines
6.8 KiB
PHP
202 lines
6.8 KiB
PHP
<?php
|
|
|
|
namespace Doctrine\DBAL\Migrations;
|
|
|
|
use Doctrine\DBAL\Migrations\Configuration\Configuration;
|
|
use Doctrine\DBAL\Migrations\Event\MigrationsEventArgs;
|
|
use const COUNT_RECURSIVE;
|
|
|
|
/**
|
|
* Class for running migrations to the current version or a manually specified version.
|
|
*
|
|
* @license http://www.opensource.org/licenses/lgpl-license.php LGPL
|
|
* @link www.doctrine-project.org
|
|
* @since 2.0
|
|
* @author Jonathan H. Wage <jonwage@gmail.com>
|
|
*/
|
|
class Migration
|
|
{
|
|
/**
|
|
* The OutputWriter object instance used for outputting information
|
|
*
|
|
* @var OutputWriter
|
|
*/
|
|
private $outputWriter;
|
|
|
|
/**
|
|
* @var Configuration
|
|
*/
|
|
private $configuration;
|
|
|
|
/**
|
|
* @var boolean
|
|
*/
|
|
private $noMigrationException;
|
|
|
|
/**
|
|
* Construct a Migration instance
|
|
*
|
|
* @param Configuration $configuration A migration Configuration instance
|
|
*/
|
|
public function __construct(Configuration $configuration)
|
|
{
|
|
$this->configuration = $configuration;
|
|
$this->outputWriter = $configuration->getOutputWriter();
|
|
$this->noMigrationException = false;
|
|
}
|
|
|
|
/**
|
|
* Get the array of versions and SQL queries that would be executed for
|
|
* each version but do not execute anything.
|
|
*
|
|
* @param string $to The version to migrate to.
|
|
*
|
|
* @return array $sql The array of SQL queries.
|
|
*/
|
|
public function getSql($to = null)
|
|
{
|
|
return $this->migrate($to, true);
|
|
}
|
|
|
|
/**
|
|
* Write a migration SQL file to the given path
|
|
*
|
|
* @param string $path The path to write the migration SQL file.
|
|
* @param string $to The version to migrate to.
|
|
*
|
|
* @return boolean $written
|
|
*/
|
|
public function writeSqlFile($path, $to = null)
|
|
{
|
|
$sql = $this->getSql($to);
|
|
$from = $this->configuration->getCurrentVersion();
|
|
|
|
if ($to === null) {
|
|
$to = $this->configuration->getLatestVersion();
|
|
}
|
|
|
|
$direction = $from > $to ? Version::DIRECTION_DOWN : Version::DIRECTION_UP;
|
|
|
|
$this->outputWriter->write(sprintf("-- Migrating from %s to %s\n", $from, $to));
|
|
|
|
/*
|
|
* Since the configuration object changes during the creation we cannot inject things
|
|
* properly, so I had to violate LoD here (so please, let's find a way to solve it on v2).
|
|
*/
|
|
return $this->configuration->getQueryWriter()
|
|
->write($path, $direction, $sql);
|
|
}
|
|
|
|
/**
|
|
* @param boolean $noMigrationException Throw an exception or not if no migration is found. Mostly for Continuous Integration.
|
|
*/
|
|
public function setNoMigrationException($noMigrationException = false)
|
|
{
|
|
$this->noMigrationException = $noMigrationException;
|
|
}
|
|
|
|
/**
|
|
* Run a migration to the current version or the given target version.
|
|
*
|
|
* @param string $to The version to migrate to.
|
|
* @param boolean $dryRun Whether or not to make this a dry run and not execute anything.
|
|
* @param boolean $timeAllQueries Measuring or not the execution time of each SQL query.
|
|
* @param callable|null $confirm A callback to confirm whether the migrations should be executed.
|
|
*
|
|
* @return array An array of migration sql statements. This will be empty if the the $confirm callback declines to execute the migration
|
|
*
|
|
* @throws MigrationException
|
|
*/
|
|
public function migrate($to = null, $dryRun = false, $timeAllQueries = false, callable $confirm = null)
|
|
{
|
|
/**
|
|
* If no version to migrate to is given we default to the last available one.
|
|
*/
|
|
if ($to === null) {
|
|
$to = $this->configuration->getLatestVersion();
|
|
}
|
|
|
|
$from = (string) $this->configuration->getCurrentVersion();
|
|
$to = (string) $to;
|
|
|
|
/**
|
|
* Throw an error if we can't find the migration to migrate to in the registered
|
|
* migrations.
|
|
*/
|
|
$migrations = $this->configuration->getMigrations();
|
|
if ( ! isset($migrations[$to]) && $to > 0) {
|
|
throw MigrationException::unknownMigrationVersion($to);
|
|
}
|
|
|
|
$direction = $from > $to ? Version::DIRECTION_DOWN : Version::DIRECTION_UP;
|
|
$migrationsToExecute = $this->configuration->getMigrationsToExecute($direction, $to);
|
|
|
|
/**
|
|
* If
|
|
* there are no migrations to execute
|
|
* and there are migrations,
|
|
* and the migration from and to are the same
|
|
* means we are already at the destination return an empty array()
|
|
* to signify that there is nothing left to do.
|
|
*/
|
|
if ($from === $to && empty($migrationsToExecute) && ! empty($migrations)) {
|
|
return $this->noMigrations();
|
|
}
|
|
|
|
if ( ! $dryRun && false === $this->migrationsCanExecute($confirm)) {
|
|
return [];
|
|
}
|
|
|
|
$output = $dryRun ? 'Executing dry run of migration' : 'Migrating';
|
|
$output .= ' <info>%s</info> to <comment>%s</comment> from <comment>%s</comment>';
|
|
$this->outputWriter->write(sprintf($output, $direction, $to, $from));
|
|
|
|
/**
|
|
* If there are no migrations to execute throw an exception.
|
|
*/
|
|
if (empty($migrationsToExecute) && ! $this->noMigrationException) {
|
|
throw MigrationException::noMigrationsToExecute();
|
|
} elseif (empty($migrationsToExecute)) {
|
|
return $this->noMigrations();
|
|
}
|
|
|
|
$this->configuration->dispatchEvent(
|
|
Events::onMigrationsMigrating,
|
|
new MigrationsEventArgs($this->configuration, $direction, $dryRun)
|
|
);
|
|
|
|
$sql = [];
|
|
$time = 0;
|
|
|
|
foreach ($migrationsToExecute as $version) {
|
|
$versionSql = $version->execute($direction, $dryRun, $timeAllQueries);
|
|
$sql[$version->getVersion()] = $versionSql;
|
|
$time += $version->getTime();
|
|
}
|
|
|
|
$this->configuration->dispatchEvent(
|
|
Events::onMigrationsMigrated,
|
|
new MigrationsEventArgs($this->configuration, $direction, $dryRun)
|
|
);
|
|
|
|
$this->outputWriter->write("\n <comment>------------------------</comment>\n");
|
|
$this->outputWriter->write(sprintf(" <info>++</info> finished in %ss", $time));
|
|
$this->outputWriter->write(sprintf(" <info>++</info> %s migrations executed", count($migrationsToExecute)));
|
|
$this->outputWriter->write(sprintf(" <info>++</info> %s sql queries", count($sql, COUNT_RECURSIVE) - count($sql)));
|
|
|
|
return $sql;
|
|
}
|
|
|
|
private function noMigrations() : array
|
|
{
|
|
$this->outputWriter->write('<comment>No migrations to execute.</comment>');
|
|
|
|
return [];
|
|
}
|
|
|
|
private function migrationsCanExecute(callable $confirm = null) : bool
|
|
{
|
|
return null === $confirm ? true : (bool) $confirm();
|
|
}
|
|
}
|