mirror of
https://github.com/doctrine/data-fixtures.git
synced 2026-03-23 22:32:13 +01:00
261 lines
8.0 KiB
PHP
261 lines
8.0 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Doctrine\Common\DataFixtures\Purger;
|
|
|
|
use Doctrine\Common\DataFixtures\Sorter\TopologicalSorter;
|
|
use Doctrine\DBAL\Platforms\AbstractPlatform;
|
|
use Doctrine\DBAL\Schema\Identifier;
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
use Doctrine\ORM\Mapping\ClassMetadata;
|
|
use Doctrine\ORM\Mapping\ManyToManyOwningSideMapping;
|
|
|
|
use function array_map;
|
|
use function array_reverse;
|
|
use function assert;
|
|
use function count;
|
|
use function in_array;
|
|
|
|
/**
|
|
* Class responsible for purging databases of data before reloading data fixtures.
|
|
*/
|
|
final class ORMPurger implements ORMPurgerInterface
|
|
{
|
|
public const PURGE_MODE_DELETE = 1;
|
|
public const PURGE_MODE_TRUNCATE = 2;
|
|
|
|
/**
|
|
* If the purge should be done through DELETE or TRUNCATE statements
|
|
*/
|
|
private int $purgeMode = self::PURGE_MODE_DELETE;
|
|
|
|
/**
|
|
* Table/view names to be excluded from purge
|
|
*
|
|
* @var string[]
|
|
*/
|
|
private array $excluded;
|
|
|
|
/** @var list<string>|null */
|
|
private array|null $cachedSqlStatements = null;
|
|
|
|
/**
|
|
* Construct new purger instance.
|
|
*
|
|
* @param EntityManagerInterface|null $em EntityManagerInterface instance used for persistence.
|
|
* @param string[] $excluded array of table/view names to be excluded from purge
|
|
*/
|
|
public function __construct(private EntityManagerInterface|null $em = null, array $excluded = [])
|
|
{
|
|
$this->excluded = $excluded;
|
|
}
|
|
|
|
/**
|
|
* Set the purge mode
|
|
*/
|
|
public function setPurgeMode(int $mode): void
|
|
{
|
|
$this->purgeMode = $mode;
|
|
$this->cachedSqlStatements = null;
|
|
}
|
|
|
|
/**
|
|
* Get the purge mode
|
|
*/
|
|
public function getPurgeMode(): int
|
|
{
|
|
return $this->purgeMode;
|
|
}
|
|
|
|
public function setEntityManager(EntityManagerInterface $em): void
|
|
{
|
|
$this->em = $em;
|
|
$this->cachedSqlStatements = null;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the EntityManagerInterface instance this purger instance is using.
|
|
*/
|
|
public function getObjectManager(): EntityManagerInterface
|
|
{
|
|
return $this->em;
|
|
}
|
|
|
|
public function purge(): void
|
|
{
|
|
$connection = $this->em->getConnection();
|
|
array_map([$connection, 'executeStatement'], $this->getPurgeStatements());
|
|
}
|
|
|
|
/** @return list<string> */
|
|
private function getPurgeStatements(): array
|
|
{
|
|
if ($this->cachedSqlStatements !== null) {
|
|
return $this->cachedSqlStatements;
|
|
}
|
|
|
|
$connection = $this->em->getConnection();
|
|
$classes = [];
|
|
|
|
foreach ($this->em->getMetadataFactory()->getAllMetadata() as $metadata) {
|
|
if ($metadata->isMappedSuperclass || (isset($metadata->isEmbeddedClass) && $metadata->isEmbeddedClass)) {
|
|
continue;
|
|
}
|
|
|
|
$classes[] = $metadata;
|
|
}
|
|
|
|
$commitOrder = $this->getCommitOrder($this->em, $classes);
|
|
|
|
// Get platform parameters
|
|
$platform = $connection->getDatabasePlatform();
|
|
|
|
// Drop association tables first
|
|
$orderedTables = $this->getAssociationTables($commitOrder, $platform);
|
|
|
|
// Drop tables in reverse commit order
|
|
for ($i = count($commitOrder) - 1; $i >= 0; --$i) {
|
|
$class = $commitOrder[$i];
|
|
|
|
if (
|
|
(isset($class->isEmbeddedClass) && $class->isEmbeddedClass) ||
|
|
$class->isMappedSuperclass ||
|
|
($class->isInheritanceTypeSingleTable() && $class->name !== $class->rootEntityName)
|
|
) {
|
|
continue;
|
|
}
|
|
|
|
$orderedTables[] = $this->getTableName($class, $platform);
|
|
}
|
|
|
|
$connectionConfiguration = $connection->getConfiguration();
|
|
|
|
$schemaAssetsFilter = $connectionConfiguration->getSchemaAssetsFilter()
|
|
?? static fn (): bool => true;
|
|
|
|
$this->cachedSqlStatements = [];
|
|
foreach ($orderedTables as $tbl) {
|
|
// If the table is excluded, skip it as well
|
|
if (in_array($tbl, $this->excluded)) {
|
|
continue;
|
|
}
|
|
|
|
// Support schema asset filters as presented in
|
|
if (! $schemaAssetsFilter($tbl)) {
|
|
continue;
|
|
}
|
|
|
|
if ($this->purgeMode === self::PURGE_MODE_DELETE) {
|
|
$this->cachedSqlStatements[] = $this->getDeleteFromTableSQL($tbl, $platform);
|
|
} else {
|
|
$this->cachedSqlStatements[] = $platform->getTruncateTableSQL($tbl, true);
|
|
}
|
|
}
|
|
|
|
return $this->cachedSqlStatements;
|
|
}
|
|
|
|
/**
|
|
* @param ClassMetadata[] $classes
|
|
*
|
|
* @return ClassMetadata[]
|
|
*/
|
|
private function getCommitOrder(EntityManagerInterface $em, array $classes): array
|
|
{
|
|
$sorter = new TopologicalSorter();
|
|
|
|
foreach ($classes as $class) {
|
|
if (! $sorter->hasNode($class->name)) {
|
|
$sorter->addNode($class->name, $class);
|
|
}
|
|
|
|
// $class before its parents
|
|
foreach ($class->parentClasses as $parentClass) {
|
|
$parentClass = $em->getClassMetadata($parentClass);
|
|
$parentClassName = $parentClass->getName();
|
|
|
|
if (! $sorter->hasNode($parentClassName)) {
|
|
$sorter->addNode($parentClassName, $parentClass);
|
|
}
|
|
|
|
$sorter->addDependency($class->name, $parentClassName);
|
|
}
|
|
|
|
foreach ($class->associationMappings as $assoc) {
|
|
if (! $assoc['isOwningSide']) {
|
|
continue;
|
|
}
|
|
|
|
$targetClass = $em->getClassMetadata($assoc['targetEntity']);
|
|
assert($targetClass instanceof ClassMetadata);
|
|
$targetClassName = $targetClass->getName();
|
|
|
|
if (! $sorter->hasNode($targetClassName)) {
|
|
$sorter->addNode($targetClassName, $targetClass);
|
|
}
|
|
|
|
// add dependency ($targetClass before $class)
|
|
$sorter->addDependency($targetClassName, $class->name);
|
|
|
|
// parents of $targetClass before $class, too
|
|
foreach ($targetClass->parentClasses as $parentClass) {
|
|
$parentClass = $em->getClassMetadata($parentClass);
|
|
$parentClassName = $parentClass->getName();
|
|
|
|
if (! $sorter->hasNode($parentClassName)) {
|
|
$sorter->addNode($parentClassName, $parentClass);
|
|
}
|
|
|
|
$sorter->addDependency($parentClassName, $class->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
return array_reverse($sorter->sort());
|
|
}
|
|
|
|
/**
|
|
* @param ClassMetadata[] $classes
|
|
*
|
|
* @return string[]
|
|
*/
|
|
private function getAssociationTables(array $classes, AbstractPlatform $platform): array
|
|
{
|
|
$associationTables = [];
|
|
|
|
foreach ($classes as $class) {
|
|
foreach ($class->associationMappings as $assoc) {
|
|
if (! $assoc['isOwningSide'] || $assoc['type'] !== ClassMetadata::MANY_TO_MANY) {
|
|
continue;
|
|
}
|
|
|
|
$associationTables[] = $this->getJoinTableName($assoc, $class, $platform);
|
|
}
|
|
}
|
|
|
|
return $associationTables;
|
|
}
|
|
|
|
private function getTableName(ClassMetadata $class, AbstractPlatform $platform): string
|
|
{
|
|
return $this->em->getConfiguration()->getQuoteStrategy()->getTableName($class, $platform);
|
|
}
|
|
|
|
/** @param ManyToManyOwningSideMapping|mixed[] $assoc */
|
|
private function getJoinTableName(
|
|
$assoc,
|
|
ClassMetadata $class,
|
|
AbstractPlatform $platform,
|
|
): string {
|
|
return $this->em->getConfiguration()->getQuoteStrategy()->getJoinTableName($assoc, $class, $platform);
|
|
}
|
|
|
|
private function getDeleteFromTableSQL(string $tableName, AbstractPlatform $platform): string
|
|
{
|
|
$tableIdentifier = new Identifier($tableName);
|
|
|
|
return 'DELETE FROM ' . $tableIdentifier->getQuotedName($platform);
|
|
}
|
|
}
|