. */ #namespace Doctrine\ORM\Mapping; #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 Konsta Vesterinen * @author Roman Borschel * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @version $Revision$ * @link www.doctrine-project.org * @since 2.0 */ class Doctrine_ORM_Mapping_ClassMetadataFactory { /** The targeted database platform. */ private $_targetPlatform; private $_driver; private $_cacheDriver; /** * Constructor. * Creates a new factory instance that uses the given metadata driver implementation. * * @param $driver The metadata driver to use. */ public function __construct($driver, Doctrine_DBAL_Platforms_AbstractPlatform $targetPlatform) { $this->_driver = $driver; $this->_targetPlatform = $targetPlatform; } public function setCacheDriver($cacheDriver) { $this->_cacheDriver = $cacheDriver; } 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])) { if ($this->_cacheDriver) { if ($this->_cacheDriver->contains("$className\$CLASSMETADATA")) { $this->_loadedMetadata[$className] = $this->_cacheDriver->get("$className\$CLASSMETADATA"); } else { $this->_loadMetadata($className); $this->_cacheDriver->put("$className\$CLASSMETADATA", $this->_loadedMetadata[$className]); } } 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) { $parentClass = $name; $parentClasses = array(); $loadedParentClass = false; while ($parentClass = get_parent_class($parentClass)) { if (isset($this->_loadedMetadata[$parentClass])) { $loadedParentClass = $parentClass; break; } $parentClasses[] = $parentClass; } $parentClasses = array_reverse($parentClasses); $parentClasses[] = $name; if ($loadedParentClass) { $class = $this->_loadedMetadata[$loadedParentClass]; } else { $rootClassOfHierarchy = count($parentClasses) > 0 ? array_shift($parentClasses) : $name; $class = $this->_newClassMetadataInstance($rootClassOfHierarchy); $this->_loadClassMetadata($class, $rootClassOfHierarchy); $this->_loadedMetadata[$rootClassOfHierarchy] = $class; } if (count($parentClasses) == 0) { return $class; } // load metadata of subclasses // -> child1 -> child2 -> $name // Move down the hierarchy of parent classes, starting from the topmost class $parent = $class; foreach ($parentClasses as $subclassName) { $subClass = $this->_newClassMetadataInstance($subclassName); $subClass->setInheritanceType($parent->getInheritanceType()); $subClass->setDiscriminatorMap($parent->getDiscriminatorMap()); $subClass->setDiscriminatorColumn($parent->getDiscriminatorColumn()); $this->_addInheritedFields($subClass, $parent); $this->_addInheritedRelations($subClass, $parent); $this->_loadClassMetadata($subClass, $subclassName); if ($parent->isInheritanceTypeSingleTable()) { $subClass->setTableName($parent->getTableName()); } $this->_loadedMetadata[$subclassName] = $subClass; $parent = $subClass; } } protected function _newClassMetadataInstance($className) { return new Doctrine_ORM_Mapping_ClassMetadata($className); } /** * Adds inherited fields to the subclass mapping. * * @param Doctrine\ORM\Mapping\ClassMetadata $subClass * @param Doctrine\ORM\Mapping\ClassMetadata $parentClass */ private function _addInheritedFields($subClass, $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($subClass, $parentClass) { foreach ($parentClass->getAssociationMappings() as $mapping) { $subClass->addAssociationMapping($mapping); } } /** * Loads the metadata of a specified class. * * @param Doctrine_ClassMetadata $class The container for the metadata. * @param string $name The name of the class for which the metadata will be loaded. */ private function _loadClassMetadata(Doctrine_ORM_Mapping_ClassMetadata $class, $name) { if ( ! class_exists($name) || empty($name)) { throw new Doctrine_Exception("Couldn't find class " . $name . "."); } $names = array(); $className = $name; // get parent classes //TODO: Skip Entity types MappedSuperclass/Transient do { if ($className == $name) { continue; } $names[] = $className; } while ($className = get_parent_class($className)); // save parents $class->setParentClasses($names); // load user-specified mapping metadata through the driver $this->_driver->loadMetadataForClass($name, $class); // set default table name, if necessary $tableName = $class->getTableName(); if ( ! isset($tableName)) { $class->setTableName(Doctrine::tableize($class->getClassName())); } // Complete Id generator mapping. If AUTO is specified we choose the generator // most appropriate for the target platform. if ($class->getIdGeneratorType() == Doctrine_ORM_Mapping_ClassMetadata::GENERATOR_TYPE_AUTO) { if ($this->_targetPlatform->prefersSequences()) { $class->setIdGeneratorType(Doctrine_ORM_Mapping_ClassMetadata::GENERATOR_TYPE_SEQUENCE); } else if ($this->_targetPlatform->prefersIdentityColumns()) { $class->setIdGeneratorType(Doctrine_ORM_Mapping_ClassMetadata::GENERATOR_TYPE_IDENTITY); } else { $class->setIdGeneratorType(Doctrine_ORM_Mapping_ClassMetadata::GENERATOR_TYPE_TABLE); } } return $class; } }