. */ 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 ($class->identifier as $field) { if (isset($class->associationMappings[$field]) && isset($id[$field]) && is_object($id[$field])) { $targetClassMetadata = $this->metadataFactory->getMetadataFor( $class->associationMappings[$field]['targetEntity'] ); $associatedId = $this->flattenIdentifier($targetClassMetadata, $this->unitOfWork->getEntityIdentifier($id[$field])); $flatId[$field] = implode(' ', $associatedId); } elseif (isset($class->associationMappings[$field])) { $associatedId = array(); foreach ($class->associationMappings[$field]['joinColumns'] as $joinColumn) { $associatedId[] = $id[$joinColumn['name']]; } $flatId[$field] = implode(' ', $associatedId); } else { $flatId[$field] = $id[$field]; } } return $flatId; } }