From 32b7a814e73318cb13859a826e036ebca5e6ce14 Mon Sep 17 00:00:00 2001 From: Alexander Rakushin Date: Thu, 27 Jul 2023 20:55:22 +0300 Subject: [PATCH] Add options in listeners: forceUseAttributeReader and separateXmlMapping --- CHANGELOG.md | 3 ++ src/Mapping/Driver/File.php | 52 ++++++++++++++++++++---- src/Mapping/ExtensionMetadataFactory.php | 35 +++++++++++++--- src/Mapping/MappedEventSubscriber.php | 28 ++++++++++++- 4 files changed, 103 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2a9702e5ba..add7be3f23 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -18,6 +18,9 @@ a release. --- ## [Unreleased] +### Fixed +- Mapping Driver: Configure usage separately with Doctrine `.orm.xml` -> `.gedmo.xml` files, also add an alternative option - force the use of AttributeReader and ignore the configuration of the Doctrine chain driver (#2613) + ### Added - SoftDeleteable: `Gedmo\SoftDeleteable\Event\PreSoftDeleteEventArgs` and `Gedmo\SoftDeleteable\Event\PostSoftDeleteEventArgs` classes. diff --git a/src/Mapping/Driver/File.php b/src/Mapping/Driver/File.php index 116bd1a8d0..365cefe1e6 100644 --- a/src/Mapping/Driver/File.php +++ b/src/Mapping/Driver/File.php @@ -120,20 +120,56 @@ abstract protected function _loadMappingFile($file); */ protected function _getMapping($className) { - // try loading mapping from original driver first $mapping = null; - if (null !== $this->_originalDriver) { - if ($this->_originalDriver instanceof FileDriver) { - $mapping = $this->_originalDriver->getElement($className); - } + $separatedFile = strpos($this->locator->getFileExtension(), '.gedmo') === 0; + + if($separatedFile){ + // try loading mapping from gedmo driver first + $mapping = $this->getMappingFromGedmoFileDriver($className); } - // if no mapping found try to load mapping file again + // if no mapping found try to load mapping file from original driver again if (null === $mapping) { - $yaml = $this->_loadMappingFile($this->locator->findMappingFile($className)); - $mapping = $yaml[$className]; + // read .orm.xml + $mapping = $this->getMappingFromOriginalDriver($className); + } + if (!$separatedFile && null === $mapping) { + // if no mapping found try to load mapping file again + $mapping = $this->getMappingFromGedmoFileDriver($className); + } + + return $mapping; + } + /** + * Tries to get a mapping for a given class from gedmo driver. + * + * @param string $className + * + * @return array|object|null + */ + private function getMappingFromGedmoFileDriver($className){ + if(!$this->locator->fileExists($className)){ + return null; } + $mapping = $this->_loadMappingFile($this->locator->findMappingFile($className)); + return $mapping[$className] ?? null; + } + + /** + * Tries to get a mapping for a given class from original doctrine driver. + * + * @param string $className + * + * @return array|object|null + */ + private function getMappingFromOriginalDriver($className){ + $mapping = null; + if (null !== $this->_originalDriver) { + if ($this->_originalDriver instanceof FileDriver) { + $mapping = $this->_originalDriver->getElement($className); + } + } return $mapping; } diff --git a/src/Mapping/ExtensionMetadataFactory.php b/src/Mapping/ExtensionMetadataFactory.php index 8b5ae49610..d1c9bc8206 100644 --- a/src/Mapping/ExtensionMetadataFactory.php +++ b/src/Mapping/ExtensionMetadataFactory.php @@ -14,6 +14,7 @@ use Doctrine\ODM\MongoDB\Mapping\ClassMetadata as DocumentClassMetadata; use Doctrine\ORM\Mapping\ClassMetadata as EntityClassMetadata; use Doctrine\ORM\Mapping\ClassMetadataInfo as LegacyEntityClassMetadata; +use Doctrine\ORM\Mapping\Driver\AttributeDriver; use Doctrine\Persistence\Mapping\ClassMetadata; use Doctrine\Persistence\Mapping\Driver\DefaultFileLocator; use Doctrine\Persistence\Mapping\Driver\MappingDriver; @@ -69,10 +70,22 @@ class ExtensionMetadataFactory private ?CacheItemPoolInterface $cacheItemPool = null; + /** + * Ignore doctrine driver class and force use attribute reader for gedmo properties + * @var bool + */ + private $forceUseAttributeReader; + + /** + * Search mapping in .gedmo.xml and does not use doctrine *.orm.xml or *.dcm.xml file + * @var bool + */ + private $separateXmlMapping; + /** * @param Reader|AttributeReader|object|null $annotationReader */ - public function __construct(ObjectManager $objectManager, string $extensionNamespace, ?object $annotationReader = null, ?CacheItemPoolInterface $cacheItemPool = null) + public function __construct(ObjectManager $objectManager, string $extensionNamespace, ?object $annotationReader = null, ?CacheItemPoolInterface $cacheItemPool = null, bool $forceUseAttributeReader = false, bool $separateXmlMapping = false) { if (null !== $annotationReader && !$annotationReader instanceof Reader && !$annotationReader instanceof AttributeReader) { trigger_deprecation( @@ -88,6 +101,9 @@ public function __construct(ObjectManager $objectManager, string $extensionNames $this->objectManager = $objectManager; $this->annotationReader = $annotationReader; $this->extensionNamespace = $extensionNamespace; + $this->forceUseAttributeReader = $forceUseAttributeReader; + $this->separateXmlMapping = $separateXmlMapping; + $omDriver = $objectManager->getConfiguration()->getMetadataDriverImpl(); $this->driver = $this->getDriver($omDriver); $this->cacheItemPool = $cacheItemPool; @@ -171,6 +187,10 @@ public static function getCacheId($className, $extensionNamespace) return str_replace('\\', '_', $className).'_$'.strtoupper(str_replace('\\', '_', $extensionNamespace)).'_CLASSMETADATA'; } + private function getFileExtension($fileExtension) + { + return $this->separateXmlMapping ? str_replace(['.orm.','.dcm.'], '.gedmo.', $fileExtension) : $fileExtension; + } /** * Get the extended driver instance which will * read the metadata required by extension @@ -192,11 +212,12 @@ protected function getDriver($omDriver) $driverName = substr($className, strrpos($className, '\\') + 1); if ($omDriver instanceof MappingDriverChain || 'DriverChain' === $driverName) { $driver = new Chain(); + $attributeDriver = $this->forceUseAttributeReader ? new AttributeDriver([]) : null; foreach ($omDriver->getDrivers() as $namespace => $nestedOmDriver) { - $driver->addDriver($this->getDriver($nestedOmDriver), $namespace); + $driver->addDriver($this->getDriver($attributeDriver ?? $nestedOmDriver), $namespace); } if (null !== $omDriver->getDefaultDriver()) { - $driver->setDefaultDriver($this->getDriver($omDriver->getDefaultDriver())); + $driver->setDefaultDriver($this->getDriver($attributeDriver ?? $omDriver->getDefaultDriver())); } } else { $driverName = substr($driverName, 0, strpos($driverName, 'Driver')); @@ -230,12 +251,14 @@ protected function getDriver($omDriver) $driver->setOriginalDriver($omDriver); if ($driver instanceof FileDriver) { if ($omDriver instanceof MappingDriver) { - $driver->setLocator($omDriver->getLocator()); + $locator = clone $omDriver->getLocator(); + $locator->setFileExtension($this->getFileExtension( $locator->getFileExtension())); + $driver->setLocator($locator); // BC for Doctrine 2.2 } elseif ($isSimplified) { - $driver->setLocator(new SymfonyFileLocator($omDriver->getNamespacePrefixes(), $omDriver->getFileExtension())); + $driver->setLocator(new SymfonyFileLocator($omDriver->getNamespacePrefixes(), $this->getFileExtension( $omDriver->getFileExtension()))); } else { - $driver->setLocator(new DefaultFileLocator($omDriver->getPaths(), $omDriver->getFileExtension())); + $driver->setLocator(new DefaultFileLocator($omDriver->getPaths(), $this->getFileExtension( $omDriver->getFileExtension()))); } } diff --git a/src/Mapping/MappedEventSubscriber.php b/src/Mapping/MappedEventSubscriber.php index 5ba286e3ee..8d17d24cb4 100644 --- a/src/Mapping/MappedEventSubscriber.php +++ b/src/Mapping/MappedEventSubscriber.php @@ -94,12 +94,36 @@ abstract class MappedEventSubscriber implements EventSubscriber */ private $cacheItemPool; + + /** + * Ignore doctrine driver class and force use attribute reader for gedmo properties + * @var bool + */ + private $forceUseAttributeReader = false; + + /** + * Search mapping in .gedmo.xml and does not use doctrine *.orm.xml or *.dcm.xml file + * @var bool + */ + private $separateXmlMapping = false; + + public function __construct() { $parts = explode('\\', $this->getNamespace()); $this->name = end($parts); } + + public function setForceUseAttributeReader(bool $forceUseAttributeReader) { + $this->forceUseAttributeReader = $forceUseAttributeReader; + } + public function setSeparateXmlMapping(bool $separateXmlMapping) { + $this->separateXmlMapping = $separateXmlMapping; + } + + + /** * Get the configuration for specific object class * if cache driver is present it scans it also @@ -178,7 +202,9 @@ public function getExtensionMetadataFactory(ObjectManager $objectManager) $objectManager, $this->getNamespace(), $this->annotationReader, - $this->getCacheItemPool($objectManager) + $this->getCacheItemPool($objectManager), + $this->forceUseAttributeReader, + $this->separateXmlMapping ); }