. */ namespace Doctrine\ORM\Mapping; use Doctrine\Common\DoctrineException; use Doctrine\DBAL\Platforms\AbstractPlatform; /** * The metadata factory is used to create ClassMetadata objects that contain all the * metadata mapping informations of a class which describes how a class should be mapped * to a relational database. * * @author Roman Borschel * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @version $Revision$ * @link www.doctrine-project.org * @since 2.0 */ class ClassMetadataFactory { /** The targeted database platform. */ private $_targetPlatform; /** The used metadata driver. */ private $_driver; /** The used cache driver. */ private $_cacheDriver; /** * Creates a new factory instance that uses the given metadata driver implementation. * * @param $driver The metadata driver to use. */ public function __construct($driver, AbstractPlatform $targetPlatform) { $this->_driver = $driver; $this->_targetPlatform = $targetPlatform; } /** * Sets the cache driver used by the factory to cache ClassMetadata instances. * * @param Doctrine\ORM\Cache\Cache $cacheDriver */ public function setCacheDriver($cacheDriver) { $this->_cacheDriver = $cacheDriver; } /** * Gets the cache driver used by the factory to cache ClassMetadata instances. * * @return Doctrine\ORM\Cache\Cache */ public function getCacheDriver() { return $this->_cacheDriver; } /** * Returns the metadata object for a class. * * @param string $className The name of the class. * @return Doctrine\ORM\Mapping\ClassMetadata */ public function getMetadataFor($className) { if ( ! isset($this->_loadedMetadata[$className])) { $cacheKey = "$className\$CLASSMETADATA"; if ($this->_cacheDriver) { if ($this->_cacheDriver->contains($cacheKey)) { $this->_loadedMetadata[$className] = $this->_cacheDriver->fetch($cacheKey); } else { $this->_loadMetadata($className); $this->_cacheDriver->save($cacheKey, $this->_loadedMetadata[$className], null); } } else { $this->_loadMetadata($className); } } return $this->_loadedMetadata[$className]; } /** * Loads the metadata of the class in question and all it's ancestors whose metadata * is still not loaded. * * @param string $name The name of the class for which the metadata should get loaded. * @param array $tables The metadata collection to which the loaded metadata is added. */ protected function _loadMetadata($name) { // Collect parent classes, ignoring transient (not-mapped) classes. $parentClass = $name; $parentClasses = array(); while ($parentClass = get_parent_class($parentClass)) { if ( ! $this->_driver->isTransient($parentClass)) { $parentClasses[] = $parentClass; } } $parentClasses = array_reverse($parentClasses); $parentClasses[] = $name; // Move down the hierarchy of parent classes, starting from the topmost class $parent = null; $visited = array(); foreach ($parentClasses as $className) { $class = $this->_newClassMetadataInstance($className); if ($parent) { $class->setInheritanceType($parent->getInheritanceType()); $class->setDiscriminatorMap($parent->getDiscriminatorMap()); $class->setDiscriminatorColumn($parent->getDiscriminatorColumn()); $class->setIdGeneratorType($parent->getIdGeneratorType()); $this->_addInheritedFields($class, $parent); $this->_addInheritedRelations($class, $parent); } // Invoke driver $this->_driver->loadMetadataForClass($className, $class); // Verify & complete identifier mapping if ( ! $class->getIdentifier()) { throw MappingException::identifierRequired($className); } $this->_completeIdGeneratorMapping($class); if ($parent && $parent->isInheritanceTypeSingleTable()) { $class->setTableName($parent->getTableName()); } $this->_loadedMetadata[$className] = $class; $parent = $class; $class->setParentClasses($visited); array_unshift($visited, $className); } } /** * Creates a new ClassMetadata instance for the given class name. * * @param string $className * @return Doctrine\ORM\Mapping\ClassMetadata */ protected function _newClassMetadataInstance($className) { return new ClassMetadata($className); } /** * Adds inherited fields to the subclass mapping. * * @param Doctrine\ORM\Mapping\ClassMetadata $subClass * @param Doctrine\ORM\Mapping\ClassMetadata $parentClass */ private function _addInheritedFields(ClassMetadata $subClass, ClassMetadata $parentClass) { foreach ($parentClass->getFieldMappings() as $fieldName => $mapping) { if ( ! isset($mapping['inherited'])) { $mapping['inherited'] = $parentClass->getClassName(); } $subClass->mapField($mapping); } } /** * Adds inherited associations to the subclass mapping. * * @param Doctrine\ORM\Mapping\ClassMetadata $subClass * @param Doctrine\ORM\Mapping\ClassMetadata $parentClass */ private function _addInheritedRelations(ClassMetadata $subClass, ClassMetadata $parentClass) { foreach ($parentClass->getAssociationMappings() as $mapping) { $subClass->addAssociationMapping($mapping); } } /** * Completes the ID generator mapping. If "auto" is specified we choose the generator * most appropriate for the targeted database platform. * * @param Doctrine\ORM\Mapping\ClassMetadata $class */ private function _completeIdGeneratorMapping(ClassMetadata $class) { $idGenType = $class->getIdGeneratorType(); if ($idGenType == ClassMetadata::GENERATOR_TYPE_AUTO) { if ($this->_targetPlatform->prefersSequences()) { $class->setIdGeneratorType(ClassMetadata::GENERATOR_TYPE_SEQUENCE); } else if ($this->_targetPlatform->prefersIdentityColumns()) { $class->setIdGeneratorType(ClassMetadata::GENERATOR_TYPE_IDENTITY); } else { $class->setIdGeneratorType(ClassMetadata::GENERATOR_TYPE_TABLE); } } // Create & assign an appropriate ID generator instance switch ($class->getIdGeneratorType()) { case ClassMetadata::GENERATOR_TYPE_IDENTITY: $class->setIdGenerator(new \Doctrine\ORM\Id\IdentityGenerator()); break; case ClassMetadata::GENERATOR_TYPE_SEQUENCE: // If there is no sequence definition yet, create a default definition $definition = $class->getSequenceGeneratorDefinition(); if ( ! $definition) { $definition['sequenceName'] = $class->getTableName() . '_' . $class->getSingleIdentifierColumnName() . '_seq'; $definition['allocationSize'] = 20; $definition['initialValue'] = 1; $class->setSequenceGeneratorDefinition($definition); } $sequenceGenerator = new \Doctrine\ORM\Id\SequenceGenerator( $definition['sequenceName'], $definition['allocationSize'] ); $class->setIdGenerator($sequenceGenerator); break; case ClassMetadata::GENERATOR_TYPE_NONE: $class->setIdGenerator(new \Doctrine\ORM\Id\Assigned()); break; case ClassMetadata::GENERATOR_TYPE_TABLE: throw new DoctrineException("DoctrineTableGenerator not yet implemented."); break; default: throw new DoctrineException("Unexhaustive match."); } } }