. */ namespace Doctrine\ORM\Utility; use Doctrine\ORM\UnitOfWork; use Doctrine\ORM\Mapping\ClassMetadata; use Doctrine\Common\Persistence\Mapping\ClassMetadataFactory; /** * The IdentifierFlattener utility now houses some of the identifier manipulation logic from unit of work, so that it * can be re-used elsewhere. * * @since 2.5 * @author Rob Caiger */ final class IdentifierFlattener { /** * The UnitOfWork used to coordinate object-level transactions. * * @var UnitOfWork */ private $unitOfWork; /** * The metadata factory, used to retrieve the ORM metadata of entity classes. * * @var ClassMetadataFactory */ private $metadataFactory; /** * Initializes a new IdentifierFlattener instance, bound to the given EntityManager. * * @param UnitOfWork $unitOfWork * @param ClassMetadataFactory $metadataFactory */ public function __construct(UnitOfWork $unitOfWork, ClassMetadataFactory $metadataFactory) { $this->unitOfWork = $unitOfWork; $this->metadataFactory = $metadataFactory; } /** * convert foreign identifiers into scalar foreign key values to avoid object to string conversion failures. * * @param ClassMetadata $class * @param array $id * * @return array */ public function flattenIdentifier(ClassMetadata $class, array $id) { $flatId = array(); foreach ($id as $idField => $idValue) { if (isset($class->associationMappings[$idField]) && is_object($idValue)) { /* @var $targetClassMetadata ClassMetadata */ $targetClassMetadata = $this->metadataFactory->getMetadataFor( $class->associationMappings[$idField]['targetEntity'] ); $associatedId = $this->unitOfWork->isInIdentityMap($idValue) ? $this->unitOfWork->getEntityIdentifier($idValue) : $targetClassMetadata->getIdentifierValues($idValue); $flatId[$idField] = $associatedId[$targetClassMetadata->identifier[0]]; } else { $flatId[$idField] = $idValue; } } return $flatId; } }