466 lines
16 KiB
PHP
466 lines
16 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Doctrine\Migrations;
|
|
|
|
use Doctrine\Common\EventManager;
|
|
use Doctrine\DBAL\Connection;
|
|
use Doctrine\Migrations\Configuration\Configuration;
|
|
use Doctrine\Migrations\Configuration\Connection\ConnectionLoader;
|
|
use Doctrine\Migrations\Configuration\EntityManager\EntityManagerLoader;
|
|
use Doctrine\Migrations\Configuration\Migration\ConfigurationLoader;
|
|
use Doctrine\Migrations\Exception\FrozenDependencies;
|
|
use Doctrine\Migrations\Exception\MissingDependency;
|
|
use Doctrine\Migrations\Finder\GlobFinder;
|
|
use Doctrine\Migrations\Finder\MigrationFinder;
|
|
use Doctrine\Migrations\Finder\RecursiveRegexFinder;
|
|
use Doctrine\Migrations\Generator\ClassNameGenerator;
|
|
use Doctrine\Migrations\Generator\ConcatenationFileBuilder;
|
|
use Doctrine\Migrations\Generator\DiffGenerator;
|
|
use Doctrine\Migrations\Generator\FileBuilder;
|
|
use Doctrine\Migrations\Generator\Generator;
|
|
use Doctrine\Migrations\Generator\SqlGenerator;
|
|
use Doctrine\Migrations\Metadata\Storage\MetadataStorage;
|
|
use Doctrine\Migrations\Metadata\Storage\TableMetadataStorage;
|
|
use Doctrine\Migrations\Metadata\Storage\TableMetadataStorageConfiguration;
|
|
use Doctrine\Migrations\Provider\DBALSchemaDiffProvider;
|
|
use Doctrine\Migrations\Provider\EmptySchemaProvider;
|
|
use Doctrine\Migrations\Provider\LazySchemaDiffProvider;
|
|
use Doctrine\Migrations\Provider\OrmSchemaProvider;
|
|
use Doctrine\Migrations\Provider\SchemaDiffProvider;
|
|
use Doctrine\Migrations\Provider\SchemaProvider;
|
|
use Doctrine\Migrations\Tools\Console\ConsoleInputMigratorConfigurationFactory;
|
|
use Doctrine\Migrations\Tools\Console\Helper\MigrationStatusInfosHelper;
|
|
use Doctrine\Migrations\Tools\Console\MigratorConfigurationFactory;
|
|
use Doctrine\Migrations\Version\AliasResolver;
|
|
use Doctrine\Migrations\Version\AlphabeticalComparator;
|
|
use Doctrine\Migrations\Version\Comparator;
|
|
use Doctrine\Migrations\Version\CurrentMigrationStatusCalculator;
|
|
use Doctrine\Migrations\Version\DbalExecutor;
|
|
use Doctrine\Migrations\Version\DbalMigrationFactory;
|
|
use Doctrine\Migrations\Version\DefaultAliasResolver;
|
|
use Doctrine\Migrations\Version\Executor;
|
|
use Doctrine\Migrations\Version\MigrationFactory;
|
|
use Doctrine\Migrations\Version\MigrationPlanCalculator;
|
|
use Doctrine\Migrations\Version\MigrationStatusCalculator;
|
|
use Doctrine\Migrations\Version\SortedMigrationPlanCalculator;
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
use Psr\Log\LoggerInterface;
|
|
use Psr\Log\NullLogger;
|
|
use Symfony\Component\Stopwatch\Stopwatch;
|
|
|
|
use function array_key_exists;
|
|
use function call_user_func;
|
|
use function method_exists;
|
|
use function preg_quote;
|
|
use function sprintf;
|
|
|
|
/**
|
|
* The DependencyFactory is responsible for wiring up and managing internal class dependencies.
|
|
*/
|
|
class DependencyFactory
|
|
{
|
|
/** @var array<string, bool> */
|
|
private array $inResolution = [];
|
|
|
|
private Configuration|null $configuration = null;
|
|
|
|
/** @var object[]|callable[] */
|
|
private array $dependencies = [];
|
|
|
|
private Connection|null $connection = null;
|
|
private EntityManagerInterface|null $em = null;
|
|
private EventManager|null $eventManager = null;
|
|
private bool $frozen = false;
|
|
private ConfigurationLoader $configurationLoader;
|
|
private ConnectionLoader $connectionLoader;
|
|
private EntityManagerLoader|null $emLoader = null;
|
|
|
|
/** @var callable[] */
|
|
private array $factories = [];
|
|
|
|
public static function fromConnection(
|
|
ConfigurationLoader $configurationLoader,
|
|
ConnectionLoader $connectionLoader,
|
|
LoggerInterface|null $logger = null,
|
|
): self {
|
|
$dependencyFactory = new self($logger);
|
|
$dependencyFactory->configurationLoader = $configurationLoader;
|
|
$dependencyFactory->connectionLoader = $connectionLoader;
|
|
|
|
return $dependencyFactory;
|
|
}
|
|
|
|
public static function fromEntityManager(
|
|
ConfigurationLoader $configurationLoader,
|
|
EntityManagerLoader $emLoader,
|
|
LoggerInterface|null $logger = null,
|
|
): self {
|
|
$dependencyFactory = new self($logger);
|
|
$dependencyFactory->configurationLoader = $configurationLoader;
|
|
$dependencyFactory->emLoader = $emLoader;
|
|
|
|
return $dependencyFactory;
|
|
}
|
|
|
|
private function __construct(LoggerInterface|null $logger)
|
|
{
|
|
if ($logger === null) {
|
|
return;
|
|
}
|
|
|
|
$this->setDefinition(LoggerInterface::class, static fn (): LoggerInterface => $logger);
|
|
}
|
|
|
|
public function isFrozen(): bool
|
|
{
|
|
return $this->frozen;
|
|
}
|
|
|
|
public function freeze(): void
|
|
{
|
|
$this->frozen = true;
|
|
}
|
|
|
|
private function assertNotFrozen(): void
|
|
{
|
|
if ($this->frozen) {
|
|
throw FrozenDependencies::new();
|
|
}
|
|
}
|
|
|
|
public function hasEntityManager(): bool
|
|
{
|
|
return $this->emLoader !== null;
|
|
}
|
|
|
|
public function setConfigurationLoader(ConfigurationLoader $configurationLoader): void
|
|
{
|
|
$this->assertNotFrozen();
|
|
$this->configurationLoader = $configurationLoader;
|
|
}
|
|
|
|
public function getConfiguration(): Configuration
|
|
{
|
|
if ($this->configuration === null) {
|
|
$this->configuration = $this->configurationLoader->getConfiguration();
|
|
$this->freeze();
|
|
}
|
|
|
|
return $this->configuration;
|
|
}
|
|
|
|
public function getConnection(): Connection
|
|
{
|
|
if ($this->connection === null) {
|
|
$this->connection = $this->hasEntityManager()
|
|
? $this->getEntityManager()->getConnection()
|
|
: $this->connectionLoader->getConnection($this->getConfiguration()->getConnectionName());
|
|
$this->freeze();
|
|
}
|
|
|
|
return $this->connection;
|
|
}
|
|
|
|
public function getEntityManager(): EntityManagerInterface
|
|
{
|
|
if ($this->em === null) {
|
|
if ($this->emLoader === null) {
|
|
throw MissingDependency::noEntityManager();
|
|
}
|
|
|
|
$this->em = $this->emLoader->getEntityManager($this->getConfiguration()->getEntityManagerName());
|
|
$this->freeze();
|
|
}
|
|
|
|
return $this->em;
|
|
}
|
|
|
|
public function getVersionComparator(): Comparator
|
|
{
|
|
return $this->getDependency(Comparator::class, static fn (): AlphabeticalComparator => new AlphabeticalComparator());
|
|
}
|
|
|
|
public function getLogger(): LoggerInterface
|
|
{
|
|
return $this->getDependency(LoggerInterface::class, static fn (): LoggerInterface => new NullLogger());
|
|
}
|
|
|
|
public function getEventDispatcher(): EventDispatcher
|
|
{
|
|
return $this->getDependency(EventDispatcher::class, fn (): EventDispatcher => new EventDispatcher(
|
|
$this->getConnection(),
|
|
$this->getEventManager(),
|
|
));
|
|
}
|
|
|
|
public function getClassNameGenerator(): ClassNameGenerator
|
|
{
|
|
return $this->getDependency(ClassNameGenerator::class, static fn (): ClassNameGenerator => new ClassNameGenerator());
|
|
}
|
|
|
|
public function getSchemaDumper(): SchemaDumper
|
|
{
|
|
return $this->getDependency(SchemaDumper::class, function (): SchemaDumper {
|
|
$excludedTables = [];
|
|
|
|
$metadataConfig = $this->getConfiguration()->getMetadataStorageConfiguration();
|
|
if ($metadataConfig instanceof TableMetadataStorageConfiguration) {
|
|
$excludedTables[] = sprintf('/^%s$/', preg_quote($metadataConfig->getTableName(), '/'));
|
|
}
|
|
|
|
return new SchemaDumper(
|
|
$this->getConnection()->getDatabasePlatform(),
|
|
$this->getConnection()->createSchemaManager(),
|
|
$this->getMigrationGenerator(),
|
|
$this->getMigrationSqlGenerator(),
|
|
$excludedTables,
|
|
);
|
|
});
|
|
}
|
|
|
|
private function getEmptySchemaProvider(): SchemaProvider
|
|
{
|
|
return $this->getDependency(EmptySchemaProvider::class, fn (): SchemaProvider => new EmptySchemaProvider($this->connection->createSchemaManager()));
|
|
}
|
|
|
|
public function hasSchemaProvider(): bool
|
|
{
|
|
try {
|
|
$this->getSchemaProvider();
|
|
} catch (MissingDependency) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public function getSchemaProvider(): SchemaProvider
|
|
{
|
|
return $this->getDependency(SchemaProvider::class, function (): SchemaProvider {
|
|
if ($this->hasEntityManager()) {
|
|
return new OrmSchemaProvider($this->getEntityManager());
|
|
}
|
|
|
|
throw MissingDependency::noSchemaProvider();
|
|
});
|
|
}
|
|
|
|
public function getDiffGenerator(): DiffGenerator
|
|
{
|
|
return $this->getDependency(DiffGenerator::class, fn (): DiffGenerator => new DiffGenerator(
|
|
$this->getConnection()->getConfiguration(),
|
|
$this->getConnection()->createSchemaManager(),
|
|
$this->getSchemaProvider(),
|
|
$this->getConnection()->getDatabasePlatform(),
|
|
$this->getMigrationGenerator(),
|
|
$this->getMigrationSqlGenerator(),
|
|
$this->getEmptySchemaProvider(),
|
|
));
|
|
}
|
|
|
|
public function getSchemaDiffProvider(): SchemaDiffProvider
|
|
{
|
|
return $this->getDependency(SchemaDiffProvider::class, fn (): LazySchemaDiffProvider => new LazySchemaDiffProvider(
|
|
new DBALSchemaDiffProvider(
|
|
$this->getConnection()->createSchemaManager(),
|
|
$this->getConnection()->getDatabasePlatform(),
|
|
),
|
|
));
|
|
}
|
|
|
|
private function getFileBuilder(): FileBuilder
|
|
{
|
|
return $this->getDependency(FileBuilder::class, static fn (): FileBuilder => new ConcatenationFileBuilder());
|
|
}
|
|
|
|
private function getParameterFormatter(): ParameterFormatter
|
|
{
|
|
return $this->getDependency(ParameterFormatter::class, fn (): ParameterFormatter => new InlineParameterFormatter($this->getConnection()));
|
|
}
|
|
|
|
public function getMigrationsFinder(): MigrationFinder
|
|
{
|
|
return $this->getDependency(MigrationFinder::class, function (): MigrationFinder {
|
|
$configs = $this->getConfiguration();
|
|
$needsRecursiveFinder = $configs->areMigrationsOrganizedByYear() || $configs->areMigrationsOrganizedByYearAndMonth();
|
|
|
|
return $needsRecursiveFinder ? new RecursiveRegexFinder() : new GlobFinder();
|
|
});
|
|
}
|
|
|
|
public function getMigrationRepository(): MigrationsRepository
|
|
{
|
|
return $this->getDependency(MigrationsRepository::class, fn (): MigrationsRepository => new FilesystemMigrationsRepository(
|
|
$this->getConfiguration()->getMigrationClasses(),
|
|
$this->getConfiguration()->getMigrationDirectories(),
|
|
$this->getMigrationsFinder(),
|
|
$this->getMigrationFactory(),
|
|
));
|
|
}
|
|
|
|
public function getMigrationFactory(): MigrationFactory
|
|
{
|
|
return $this->getDependency(MigrationFactory::class, fn (): MigrationFactory => new DbalMigrationFactory($this->getConnection(), $this->getLogger()));
|
|
}
|
|
|
|
public function setService(string $id, object|callable $service): void
|
|
{
|
|
$this->assertNotFrozen();
|
|
$this->dependencies[$id] = $service;
|
|
}
|
|
|
|
public function getMetadataStorage(): MetadataStorage
|
|
{
|
|
return $this->getDependency(MetadataStorage::class, fn (): MetadataStorage => new TableMetadataStorage(
|
|
$this->getConnection(),
|
|
$this->getVersionComparator(),
|
|
$this->getConfiguration()->getMetadataStorageConfiguration(),
|
|
$this->getMigrationRepository(),
|
|
));
|
|
}
|
|
|
|
private function getVersionExecutor(): Executor
|
|
{
|
|
return $this->getDependency(Executor::class, fn (): Executor => new DbalExecutor(
|
|
$this->getMetadataStorage(),
|
|
$this->getEventDispatcher(),
|
|
$this->getConnection(),
|
|
$this->getSchemaDiffProvider(),
|
|
$this->getLogger(),
|
|
$this->getParameterFormatter(),
|
|
$this->getStopwatch(),
|
|
));
|
|
}
|
|
|
|
public function getQueryWriter(): QueryWriter
|
|
{
|
|
return $this->getDependency(QueryWriter::class, fn (): QueryWriter => new FileQueryWriter(
|
|
$this->getFileBuilder(),
|
|
$this->getLogger(),
|
|
));
|
|
}
|
|
|
|
public function getVersionAliasResolver(): AliasResolver
|
|
{
|
|
return $this->getDependency(AliasResolver::class, fn (): AliasResolver => new DefaultAliasResolver(
|
|
$this->getMigrationPlanCalculator(),
|
|
$this->getMetadataStorage(),
|
|
$this->getMigrationStatusCalculator(),
|
|
));
|
|
}
|
|
|
|
public function getMigrationStatusCalculator(): MigrationStatusCalculator
|
|
{
|
|
return $this->getDependency(MigrationStatusCalculator::class, fn (): MigrationStatusCalculator => new CurrentMigrationStatusCalculator(
|
|
$this->getMigrationPlanCalculator(),
|
|
$this->getMetadataStorage(),
|
|
));
|
|
}
|
|
|
|
public function getMigrationPlanCalculator(): MigrationPlanCalculator
|
|
{
|
|
return $this->getDependency(MigrationPlanCalculator::class, fn (): MigrationPlanCalculator => new SortedMigrationPlanCalculator(
|
|
$this->getMigrationRepository(),
|
|
$this->getMetadataStorage(),
|
|
$this->getVersionComparator(),
|
|
));
|
|
}
|
|
|
|
public function getMigrationGenerator(): Generator
|
|
{
|
|
return $this->getDependency(Generator::class, fn (): Generator => new Generator($this->getConfiguration()));
|
|
}
|
|
|
|
public function getMigrationSqlGenerator(): SqlGenerator
|
|
{
|
|
return $this->getDependency(SqlGenerator::class, fn (): SqlGenerator => new SqlGenerator(
|
|
$this->getConfiguration(),
|
|
$this->getConnection()->getDatabasePlatform(),
|
|
));
|
|
}
|
|
|
|
public function getConsoleInputMigratorConfigurationFactory(): MigratorConfigurationFactory
|
|
{
|
|
return $this->getDependency(MigratorConfigurationFactory::class, fn (): MigratorConfigurationFactory => new ConsoleInputMigratorConfigurationFactory(
|
|
$this->getConfiguration(),
|
|
));
|
|
}
|
|
|
|
public function getMigrationStatusInfosHelper(): MigrationStatusInfosHelper
|
|
{
|
|
return $this->getDependency(MigrationStatusInfosHelper::class, fn (): MigrationStatusInfosHelper => new MigrationStatusInfosHelper(
|
|
$this->getConfiguration(),
|
|
$this->getConnection(),
|
|
$this->getVersionAliasResolver(),
|
|
$this->getMigrationPlanCalculator(),
|
|
$this->getMigrationStatusCalculator(),
|
|
$this->getMetadataStorage(),
|
|
));
|
|
}
|
|
|
|
public function getMigrator(): Migrator
|
|
{
|
|
return $this->getDependency(Migrator::class, fn (): Migrator => new DbalMigrator(
|
|
$this->getConnection(),
|
|
$this->getEventDispatcher(),
|
|
$this->getVersionExecutor(),
|
|
$this->getLogger(),
|
|
$this->getStopwatch(),
|
|
));
|
|
}
|
|
|
|
public function getStopwatch(): Stopwatch
|
|
{
|
|
return $this->getDependency(Stopwatch::class, static fn (): Stopwatch => new Stopwatch(true));
|
|
}
|
|
|
|
public function getRollup(): Rollup
|
|
{
|
|
return $this->getDependency(Rollup::class, fn (): Rollup => new Rollup(
|
|
$this->getMetadataStorage(),
|
|
$this->getMigrationRepository(),
|
|
));
|
|
}
|
|
|
|
private function getDependency(string $id, callable $callback): mixed
|
|
{
|
|
if (! isset($this->inResolution[$id]) && array_key_exists($id, $this->factories) && ! array_key_exists($id, $this->dependencies)) {
|
|
$this->inResolution[$id] = true;
|
|
$this->dependencies[$id] = call_user_func($this->factories[$id], $this);
|
|
unset($this->inResolution);
|
|
}
|
|
|
|
if (! array_key_exists($id, $this->dependencies)) {
|
|
$this->dependencies[$id] = $callback();
|
|
}
|
|
|
|
return $this->dependencies[$id];
|
|
}
|
|
|
|
public function setDefinition(string $id, callable $service): void
|
|
{
|
|
$this->assertNotFrozen();
|
|
$this->factories[$id] = $service;
|
|
}
|
|
|
|
private function getEventManager(): EventManager
|
|
{
|
|
if ($this->eventManager !== null) {
|
|
return $this->eventManager;
|
|
}
|
|
|
|
if ($this->hasEntityManager()) {
|
|
return $this->eventManager = $this->getEntityManager()->getEventManager();
|
|
}
|
|
|
|
if (method_exists(Connection::class, 'getEventManager')) {
|
|
// DBAL < 4
|
|
return $this->eventManager = $this->getConnection()->getEventManager();
|
|
}
|
|
|
|
return $this->eventManager = new EventManager();
|
|
}
|
|
}
|