. */ namespace Doctrine\ORM\Mapping\Driver; use Doctrine\Common\DoctrineException, Doctrine\Common\Cache\ArrayCache, Doctrine\Common\Annotations\AnnotationReader, Doctrine\DBAL\Schema\AbstractSchemaManager, Doctrine\ORM\Mapping\ClassMetadataInfo, Doctrine\ORM\Mapping\MappingException, Doctrine\Common\Util\Inflector; /** * The DatabaseDriver reverse engineers the mapping metadata from a database * * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @link www.doctrine-project.org * @since 2.0 * @version $Revision$ * @author Guilherme Blanco * @author Jonathan Wage * @author Roman Borschel */ class DatabaseDriver implements Driver { /** The SchemaManager. */ private $_sm; /** * Initializes a new AnnotationDriver that uses the given AnnotationReader for reading * docblock annotations. * * @param AnnotationReader $reader The AnnotationReader to use. */ public function __construct(AbstractSchemaManager $schemaManager) { $this->_sm = $schemaManager; } /** * {@inheritdoc} */ public function loadMetadataForClass($className, ClassMetadataInfo $metadata) { $tableName = $className; $className = Inflector::classify($tableName); $metadata->name = $className; $metadata->primaryTable['name'] = $tableName; $columns = $this->_sm->listTableColumns($tableName); if($this->_sm->getDatabasePlatform()->supportsForeignKeyConstraints()) { $foreignKeys = $this->_sm->listTableForeignKeys($tableName); } else { $foreignKeys = array(); } $indexes = $this->_sm->listTableIndexes($tableName); $ids = array(); $fieldMappings = array(); foreach ($columns as $column) { // Skip columns that are foreign keys foreach ($foreignKeys as $foreignKey) { if (in_array(strtolower($column->getName()), array_map('strtolower', $foreignKey->getColumns()))) { continue(2); } } $fieldMapping = array(); if (in_array($column->getName(), $indexes['primary']->getColumns())) { $fieldMapping['id'] = true; } $fieldMapping['fieldName'] = Inflector::camelize($column->getName()); $fieldMapping['columnName'] = $column->getName(); $fieldMapping['type'] = strtolower((string) $column->getType()); if ($column->getType() instanceof \Doctrine\DBAL\Types\StringType) { $fieldMapping['length'] = $column->getLength(); $fieldMapping['fixed'] = $column->getFixed(); } else if ($column->getType() instanceof \Doctrine\DBAL\Types\IntegerType) { $fieldMapping['unsigned'] = $column->getUnsigned(); } $fieldMapping['notnull'] = $column->getNotNull(); $fieldMapping['default'] = $column->getDefault(); if (isset($fieldMapping['id'])) { $ids[] = $fieldMapping; } else { $fieldMappings[] = $fieldMapping; } } if ($ids) { if (count($ids) == 1) { $metadata->setIdGeneratorType(ClassMetadataInfo::GENERATOR_TYPE_AUTO); } foreach ($ids as $id) { $metadata->mapField($id); } } foreach ($fieldMappings as $fieldMapping) { $metadata->mapField($fieldMapping); } foreach ($foreignKeys as $foreignKey) { if (count($foreignKey->getColumns()) != 1) { throw new MappingException( "Cannot generate mapping for table '".$tableName."' with foreign keys with multiple local columns." ); } $localColumn = current($foreignKey->getColumns()); if (count($foreignKey->getForeignColumns()) != 1) { throw new MappingException( "Cannot generate mapping for table '".$tableName."' with foreign keys with multiple foreign columns." ); } $foreignColumn = current($foreignKey->getForeignColumns()); $associationMapping = array(); $associationMapping['fieldName'] = Inflector::camelize(str_ireplace('_id', '', $localColumn)); $associationMapping['columnName'] = $localColumn; $associationMapping['targetEntity'] = Inflector::classify($foreignKey->getForeignTableName()); $associationMapping['joinColumns'][] = array( 'name' => $localColumn, 'referencedColumnName' => $foreignColumn ); $metadata->mapManyToOne($associationMapping); } } /** * Whether the class with the specified name should have its metadata loaded. * This is only the case if it is either mapped as an Entity or a * MappedSuperclass. * * @param string $className * @return boolean */ public function isTransient($className) { return true; } /** * Preloads all mapping information found in any documents within the * configured paths and returns a list of class names that have been preloaded. * * @return array The list of class names that have been preloaded. */ public function preload() { $tables = array(); foreach ($this->_sm->listTables() as $table) { $tables[] = $table->getName(); } return $tables; } }