From fb055ca75d3ee4f126f17869cc6cb645d3bb5cd2 Mon Sep 17 00:00:00 2001 From: Thomas Rothe <th.rothe@gmail.com> Date: Sun, 16 Dec 2012 18:20:10 +0100 Subject: [PATCH 1/2] fixed problems with joined inheritance and composite keys SchemaTool now creates all Id columns not just only the first one. Insert statement for child entity now contains parameter for additional key columns only once. --- .../Persisters/JoinedSubclassPersister.php | 5 +- lib/Doctrine/ORM/Tools/SchemaTool.php | 141 ++++++++++++++---- .../JoinedChildClass.php | 21 +++ .../JoinedRootClass.php | 25 ++++ .../JoinedTableCompositeKeyTest.php | 64 ++++++++ .../Doctrine/Tests/OrmFunctionalTestCase.php | 10 ++ 6 files changed, 233 insertions(+), 33 deletions(-) create mode 100644 tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedChildClass.php create mode 100644 tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedRootClass.php create mode 100644 tests/Doctrine/Tests/ORM/Functional/JoinedTableCompositeKeyTest.php diff --git a/lib/Doctrine/ORM/Persisters/JoinedSubclassPersister.php b/lib/Doctrine/ORM/Persisters/JoinedSubclassPersister.php index 58aa7df17..8ad88c1d8 100644 --- a/lib/Doctrine/ORM/Persisters/JoinedSubclassPersister.php +++ b/lib/Doctrine/ORM/Persisters/JoinedSubclassPersister.php @@ -182,6 +182,7 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister // Execute inserts on subtables. // The order doesn't matter because all child tables link to the root table via FK. foreach ($subTableStmts as $tableName => $stmt) { + /** @var \Doctrine\DBAL\Statement $stmt */ $paramIndex = 1; $data = isset($insertData[$tableName]) ? $insertData[$tableName] @@ -194,7 +195,9 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister } foreach ($data as $columnName => $value) { - $stmt->bindValue($paramIndex++, $value, $this->columnTypes[$columnName]); + if (!isset($id[$columnName])) { + $stmt->bindValue($paramIndex++, $value, $this->columnTypes[$columnName]); + } } $stmt->execute(); diff --git a/lib/Doctrine/ORM/Tools/SchemaTool.php b/lib/Doctrine/ORM/Tools/SchemaTool.php index 6a5319b1c..f917b9d06 100644 --- a/lib/Doctrine/ORM/Tools/SchemaTool.php +++ b/lib/Doctrine/ORM/Tools/SchemaTool.php @@ -91,7 +91,7 @@ class SchemaTool foreach ($createSchemaSql as $sql) { try { $conn->executeQuery($sql); - } catch(\Exception $e) { + } catch (\Exception $e) { throw ToolsException::schemaToolFailure($sql, $e); } } @@ -147,6 +147,7 @@ class SchemaTool $blacklistedFks = array(); foreach ($classes as $class) { + /** @var \Doctrine\ORM\Mapping\ClassMetadata $class */ if ($this->processingNotRequired($class, $processedClasses)) { continue; } @@ -154,7 +155,7 @@ class SchemaTool $table = $schema->createTable($this->quoteStrategy->getTableName($class, $this->platform)); if ($class->isInheritanceTypeSingleTable()) { - $columns = $this->gatherColumns($class, $table); + $this->gatherColumns($class, $table); $this->gatherRelationsSql($class, $table, $schema, $addedFks, $blacklistedFks); // Add the discriminator column @@ -177,7 +178,11 @@ class SchemaTool $pkColumns = array(); foreach ($class->fieldMappings as $fieldName => $mapping) { if ( ! isset($mapping['inherited'])) { - $columnName = $this->quoteStrategy->getColumnName($mapping['fieldName'], $class, $this->platform); + $columnName = $this->quoteStrategy->getColumnName( + $mapping['fieldName'], + $class, + $this->platform + ); $this->gatherColumn($class, $mapping, $table); if ($class->isIdentifier($fieldName)) { @@ -193,20 +198,36 @@ class SchemaTool $this->addDiscriminatorColumnDefinition($class, $table); } else { // Add an ID FK column to child tables - /* @var \Doctrine\ORM\Mapping\ClassMetadata $class */ - $idMapping = $class->fieldMappings[$class->identifier[0]]; - $this->gatherColumn($class, $idMapping, $table); - $columnName = $this->quoteStrategy->getColumnName($class->identifier[0], $class, $this->platform); - // TODO: This seems rather hackish, can we optimize it? - $table->getColumn($columnName)->setAutoincrement(false); + $inheritedKeyColumns = array(); + foreach ($class->identifier as $identifierField) { + $idMapping = $class->fieldMappings[$identifierField]; + if (isset($idMapping['inherited'])) { + $this->gatherColumn($class, $idMapping, $table); + $columnName = $this->quoteStrategy->getColumnName( + $identifierField, + $class, + $this->platform + ); + // TODO: This seems rather hackish, can we optimize it? + $table->getColumn($columnName)->setAutoincrement(false); - $pkColumns[] = $columnName; + $pkColumns[] = $columnName; + $inheritedKeyColumns[] = $columnName; + } + } + if (!empty($inheritedKeyColumns)) { + // Add a FK constraint on the ID column + $table->addForeignKeyConstraint( + $this->quoteStrategy->getTableName( + $this->em->getClassMetadata($class->rootEntityName), + $this->platform + ), + $inheritedKeyColumns, + $inheritedKeyColumns, + array('onDelete' => 'CASCADE') + ); + } - // Add a FK constraint on the ID column - $table->addForeignKeyConstraint( - $this->quoteStrategy->getTableName($this->em->getClassMetadata($class->rootEntityName), $this->platform), - array($columnName), array($columnName), array('onDelete' => 'CASCADE') - ); } $table->setPrimaryKey($pkColumns); @@ -268,7 +289,10 @@ class SchemaTool } if ($eventManager->hasListeners(ToolEvents::postGenerateSchemaTable)) { - $eventManager->dispatchEvent(ToolEvents::postGenerateSchemaTable, new GenerateSchemaTableEventArgs($class, $schema, $table)); + $eventManager->dispatchEvent( + ToolEvents::postGenerateSchemaTable, + new GenerateSchemaTableEventArgs($class, $schema, $table) + ); } } @@ -277,7 +301,10 @@ class SchemaTool } if ($eventManager->hasListeners(ToolEvents::postGenerateSchema)) { - $eventManager->dispatchEvent(ToolEvents::postGenerateSchema, new GenerateSchemaEventArgs($this->em, $schema)); + $eventManager->dispatchEvent( + ToolEvents::postGenerateSchema, + new GenerateSchemaEventArgs($this->em, $schema) + ); } return $schema; @@ -296,7 +323,9 @@ class SchemaTool { $discrColumn = $class->discriminatorColumn; - if ( ! isset($discrColumn['type']) || (strtolower($discrColumn['type']) == 'string' && $discrColumn['length'] === null)) { + if ( ! isset($discrColumn['type']) || + (strtolower($discrColumn['type']) == 'string' && $discrColumn['length'] === null) + ) { $discrColumn['type'] = 'string'; $discrColumn['length'] = 255; } @@ -339,7 +368,7 @@ class SchemaTool // For now, this is a hack required for single table inheritence, since this method is called // twice by single table inheritence relations - if(!$table->hasIndex('primary')) { + if (!$table->hasIndex('primary')) { //$table->setPrimaryKey($pkColumns); } } @@ -367,7 +396,7 @@ class SchemaTool $options['platformOptions'] = array(); $options['platformOptions']['version'] = $class->isVersioned && $class->versionField == $mapping['fieldName'] ? true : false; - if(strtolower($columnType) == 'string' && $options['length'] === null) { + if (strtolower($columnType) == 'string' && $options['length'] === null) { $options['length'] = 255; } @@ -452,9 +481,18 @@ class SchemaTool if ($mapping['type'] & ClassMetadata::TO_ONE && $mapping['isOwningSide']) { $primaryKeyColumns = $uniqueConstraints = array(); // PK is unnecessary for this relation-type - $this->_gatherRelationJoinColumns($mapping['joinColumns'], $table, $foreignClass, $mapping, $primaryKeyColumns, $uniqueConstraints, $addedFks, $blacklistedFks); + $this->gatherRelationJoinColumns( + $mapping['joinColumns'], + $table, + $foreignClass, + $mapping, + $primaryKeyColumns, + $uniqueConstraints, + $addedFks, + $blacklistedFks + ); - foreach($uniqueConstraints as $indexName => $unique) { + foreach ($uniqueConstraints as $indexName => $unique) { $table->addUniqueIndex($unique['columns'], is_numeric($indexName) ? null : $indexName); } } elseif ($mapping['type'] == ClassMetadata::ONE_TO_MANY && $mapping['isOwningSide']) { @@ -464,19 +502,39 @@ class SchemaTool // create join table $joinTable = $mapping['joinTable']; - $theJoinTable = $schema->createTable($this->quoteStrategy->getJoinTableName($mapping, $foreignClass, $this->platform)); + $theJoinTable = $schema->createTable( + $this->quoteStrategy->getJoinTableName($mapping, $foreignClass, $this->platform) + ); $primaryKeyColumns = $uniqueConstraints = array(); // Build first FK constraint (relation table => source table) - $this->_gatherRelationJoinColumns($joinTable['joinColumns'], $theJoinTable, $class, $mapping, $primaryKeyColumns, $uniqueConstraints, $addedFks, $blacklistedFks); + $this->gatherRelationJoinColumns( + $joinTable['joinColumns'], + $theJoinTable, + $class, + $mapping, + $primaryKeyColumns, + $uniqueConstraints, + $addedFks, + $blacklistedFks + ); // Build second FK constraint (relation table => target table) - $this->_gatherRelationJoinColumns($joinTable['inverseJoinColumns'], $theJoinTable, $foreignClass, $mapping, $primaryKeyColumns, $uniqueConstraints, $addedFks, $blacklistedFks); + $this->gatherRelationJoinColumns( + $joinTable['inverseJoinColumns'], + $theJoinTable, + $foreignClass, + $mapping, + $primaryKeyColumns, + $uniqueConstraints, + $addedFks, + $blacklistedFks + ); $theJoinTable->setPrimaryKey($primaryKeyColumns); - foreach($uniqueConstraints as $indexName => $unique) { + foreach ($uniqueConstraints as $indexName => $unique) { $theJoinTable->addUniqueIndex($unique['columns'], is_numeric($indexName) ? null : $indexName); } } @@ -507,7 +565,8 @@ class SchemaTool if (in_array($referencedColumnName, $class->getIdentifierColumnNames())) { // it seems to be an entity as foreign key foreach ($class->getIdentifierFieldNames() as $fieldName) { - if ($class->hasAssociation($fieldName) && $class->getSingleAssociationJoinColumnName($fieldName) == $referencedColumnName) { + if ($class->hasAssociation($fieldName) + && $class->getSingleAssociationJoinColumnName($fieldName) == $referencedColumnName) { return $this->getDefiningClass( $this->em->getClassMetadata($class->associationMappings[$fieldName]['targetEntity']), $class->getSingleAssociationReferencedJoinColumnName($fieldName) @@ -531,8 +590,16 @@ class SchemaTool * @param array $addedFks * @param array $blacklistedFks */ - private function _gatherRelationJoinColumns($joinColumns, $theJoinTable, $class, $mapping, &$primaryKeyColumns, &$uniqueConstraints, &$addedFks, &$blacklistedFks) - { + private function gatherRelationJoinColumns( + $joinColumns, + $theJoinTable, + $class, + $mapping, + &$primaryKeyColumns, + &$uniqueConstraints, + &$addedFks, + &$blacklistedFks + ) { $localColumns = array(); $foreignColumns = array(); $fkOptions = array(); @@ -540,7 +607,10 @@ class SchemaTool foreach ($joinColumns as $joinColumn) { - list($definingClass, $referencedFieldName) = $this->getDefiningClass($class, $joinColumn['referencedColumnName']); + list($definingClass, $referencedFieldName) = $this->getDefiningClass( + $class, + $joinColumn['referencedColumnName'] + ); if ( ! $definingClass) { throw new \Doctrine\ORM\ORMException( @@ -550,7 +620,11 @@ class SchemaTool } $quotedColumnName = $this->quoteStrategy->getJoinColumnName($joinColumn, $class, $this->platform); - $quotedRefColumnName = $this->quoteStrategy->getReferencedJoinColumnName($joinColumn, $class, $this->platform); + $quotedRefColumnName = $this->quoteStrategy->getReferencedJoinColumnName( + $joinColumn, + $class, + $this->platform + ); $primaryKeyColumns[] = $quotedColumnName; $localColumns[] = $quotedColumnName; @@ -617,7 +691,10 @@ class SchemaTool } elseif (!isset($blacklistedFks[$compositeName])) { $addedFks[$compositeName] = array('foreignTableName' => $foreignTableName, 'foreignColumns' => $foreignColumns); $theJoinTable->addUnnamedForeignKeyConstraint( - $foreignTableName, $localColumns, $foreignColumns, $fkOptions + $foreignTableName, + $localColumns, + $foreignColumns, + $fkOptions ); } } diff --git a/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedChildClass.php b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedChildClass.php new file mode 100644 index 000000000..ba5973647 --- /dev/null +++ b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedChildClass.php @@ -0,0 +1,21 @@ +<?php +namespace Doctrine\Tests\Models\CompositeKeyInheritance; + +/** + * @Entity + */ +class JoinedChildClass extends JoinedRootClass +{ + /** + * @var string + * @Column(type="string") + */ + public $extension = 'ext'; + + /** + * @var string + * @Column(type="string") + * @Id + */ + private $additionalId = 'additional'; +} diff --git a/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedRootClass.php b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedRootClass.php new file mode 100644 index 000000000..f145ba259 --- /dev/null +++ b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/JoinedRootClass.php @@ -0,0 +1,25 @@ +<?php +namespace Doctrine\Tests\Models\CompositeKeyInheritance; + +/** + * @Entity + * @InheritanceType("JOINED") + * @DiscriminatorColumn(name="discr", type="string") + * @DiscriminatorMap({"child" = "JoinedChildClass",}) + */ +class JoinedRootClass +{ + /** + * @var string + * @Column(type="string") + * @Id + */ + protected $keyPart1 = 'part-1'; + + /** + * @var string + * @Column(type="string") + * @Id + */ + protected $keyPart2 = 'part-2'; +} diff --git a/tests/Doctrine/Tests/ORM/Functional/JoinedTableCompositeKeyTest.php b/tests/Doctrine/Tests/ORM/Functional/JoinedTableCompositeKeyTest.php new file mode 100644 index 000000000..3bc5e25ad --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Functional/JoinedTableCompositeKeyTest.php @@ -0,0 +1,64 @@ +<?php +namespace Doctrine\Tests\ORM\Functional; + +use Doctrine\Tests\OrmFunctionalTestCase; +use Doctrine\Tests\Models\CompositeKeyInheritance\JoinedChildClass; + +class JoinedTableCompositeKeyTest extends OrmFunctionalTestCase +{ + + public function setUp() + { + $this->useModelSet('compositekeyinheritance'); + parent::setUp(); + + } + + /** + * + */ + public function testInsertWithCompositeKey() + { + $childEntity = new JoinedChildClass(); + $this->_em->persist($childEntity); + $this->_em->flush(); + + $this->_em->clear(); + + $entity = $this->findEntity(); + $this->assertEquals($childEntity, $entity); + } + + /** + * + */ + public function testUpdateWithCompositeKey() + { + $childEntity = new JoinedChildClass(); + $this->_em->persist($childEntity); + $this->_em->flush(); + + $this->_em->clear(); + + $entity = $this->findEntity(); + $entity->extension = 'ext-new'; + $this->_em->persist($entity); + $this->_em->flush(); + + $this->_em->clear(); + + $persistedEntity = $this->findEntity(); + $this->assertEquals($entity, $persistedEntity); + } + + /** + * @return \Doctrine\Tests\Models\CompositeKeyInheritance\JoinedChildClass + */ + private function findEntity() + { + return $this->_em->find( + 'Doctrine\Tests\Models\CompositeKeyInheritance\JoinedRootClass', + array('keyPart1' => 'part-1', 'keyPart2' => 'part-2') + ); + } +} diff --git a/tests/Doctrine/Tests/OrmFunctionalTestCase.php b/tests/Doctrine/Tests/OrmFunctionalTestCase.php index 4b72a9a27..872f86c81 100644 --- a/tests/Doctrine/Tests/OrmFunctionalTestCase.php +++ b/tests/Doctrine/Tests/OrmFunctionalTestCase.php @@ -124,6 +124,10 @@ abstract class OrmFunctionalTestCase extends OrmTestCase 'Doctrine\Tests\Models\CustomType\CustomTypeParent', 'Doctrine\Tests\Models\CustomType\CustomTypeUpperCase', ), + 'compositekeyinheritance' => array( + 'Doctrine\Tests\Models\CompositeKeyInheritance\JoinedRootClass', + 'Doctrine\Tests\Models\CompositeKeyInheritance\JoinedChildClass', + ) ); protected function useModelSet($setName) @@ -239,6 +243,12 @@ abstract class OrmFunctionalTestCase extends OrmTestCase $conn->executeUpdate('DELETE FROM customtype_uppercases'); } + if (isset($this->_usedModelSets['compositekeyinheritance'])) { + $conn->executeUpdate('DELETE FROM JoinedChildClass'); + $conn->executeUpdate('DELETE FROM JoinedRootClass'); + } + + $this->_em->clear(); } From 86c33d78d08afa3ed0b759884e09d497878094e5 Mon Sep 17 00:00:00 2001 From: Thomas Rothe <th.rothe@gmail.com> Date: Mon, 17 Dec 2012 11:01:20 +0100 Subject: [PATCH 2/2] inheritance with composite keys added tests for single table inheritance --- .../SingleChildClass.php | 21 ++++++ .../SingleRootClass.php | 25 ++++++++ .../SingleTableCompositeKeyTest.php | 64 +++++++++++++++++++ .../Doctrine/Tests/OrmFunctionalTestCase.php | 3 + 4 files changed, 113 insertions(+) create mode 100644 tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleChildClass.php create mode 100644 tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleRootClass.php create mode 100644 tests/Doctrine/Tests/ORM/Functional/SingleTableCompositeKeyTest.php diff --git a/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleChildClass.php b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleChildClass.php new file mode 100644 index 000000000..05579bce4 --- /dev/null +++ b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleChildClass.php @@ -0,0 +1,21 @@ +<?php +namespace Doctrine\Tests\Models\CompositeKeyInheritance; + +/** + * @Entity + */ +class SingleChildClass extends SingleRootClass +{ + /** + * @var string + * @Column(type="string") + */ + public $extension = 'ext'; + + /** + * @var string + * @Column(type="string") + * @Id + */ + private $additionalId = 'additional'; +} diff --git a/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleRootClass.php b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleRootClass.php new file mode 100644 index 000000000..1bbd09702 --- /dev/null +++ b/tests/Doctrine/Tests/Models/CompositeKeyInheritance/SingleRootClass.php @@ -0,0 +1,25 @@ +<?php +namespace Doctrine\Tests\Models\CompositeKeyInheritance; + +/** + * @Entity + * @InheritanceType("SINGLE_TABLE") + * @DiscriminatorColumn(name="discr", type="string") + * @DiscriminatorMap({"child" = "SingleChildClass",}) + */ +class SingleRootClass +{ + /** + * @var string + * @Column(type="string") + * @Id + */ + protected $keyPart1 = 'part-1'; + + /** + * @var string + * @Column(type="string") + * @Id + */ + protected $keyPart2 = 'part-2'; +} diff --git a/tests/Doctrine/Tests/ORM/Functional/SingleTableCompositeKeyTest.php b/tests/Doctrine/Tests/ORM/Functional/SingleTableCompositeKeyTest.php new file mode 100644 index 000000000..3d1fe7714 --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Functional/SingleTableCompositeKeyTest.php @@ -0,0 +1,64 @@ +<?php +namespace Doctrine\Tests\ORM\Functional; + +use Doctrine\Tests\OrmFunctionalTestCase; +use Doctrine\Tests\Models\CompositeKeyInheritance\SingleChildClass; + +class SingleTableCompositeKeyTest extends OrmFunctionalTestCase +{ + + public function setUp() + { + $this->useModelSet('compositekeyinheritance'); + parent::setUp(); + + } + + /** + * + */ + public function testInsertWithCompositeKey() + { + $childEntity = new SingleChildClass(); + $this->_em->persist($childEntity); + $this->_em->flush(); + + $this->_em->clear(); + + $entity = $this->findEntity(); + $this->assertEquals($childEntity, $entity); + } + + /** + * + */ + public function testUpdateWithCompositeKey() + { + $childEntity = new SingleChildClass(); + $this->_em->persist($childEntity); + $this->_em->flush(); + + $this->_em->clear(); + + $entity = $this->findEntity(); + $entity->extension = 'ext-new'; + $this->_em->persist($entity); + $this->_em->flush(); + + $this->_em->clear(); + + $persistedEntity = $this->findEntity(); + $this->assertEquals($entity, $persistedEntity); + } + + /** + * @return \Doctrine\Tests\Models\CompositeKeyInheritance\JoinedChildClass + */ + private function findEntity() + { + return $this->_em->find( + 'Doctrine\Tests\Models\CompositeKeyInheritance\SingleRootClass', + array('keyPart1' => 'part-1', 'keyPart2' => 'part-2') + ); + } +} diff --git a/tests/Doctrine/Tests/OrmFunctionalTestCase.php b/tests/Doctrine/Tests/OrmFunctionalTestCase.php index 872f86c81..18fd19297 100644 --- a/tests/Doctrine/Tests/OrmFunctionalTestCase.php +++ b/tests/Doctrine/Tests/OrmFunctionalTestCase.php @@ -127,6 +127,8 @@ abstract class OrmFunctionalTestCase extends OrmTestCase 'compositekeyinheritance' => array( 'Doctrine\Tests\Models\CompositeKeyInheritance\JoinedRootClass', 'Doctrine\Tests\Models\CompositeKeyInheritance\JoinedChildClass', + 'Doctrine\Tests\Models\CompositeKeyInheritance\SingleRootClass', + 'Doctrine\Tests\Models\CompositeKeyInheritance\SingleChildClass', ) ); @@ -246,6 +248,7 @@ abstract class OrmFunctionalTestCase extends OrmTestCase if (isset($this->_usedModelSets['compositekeyinheritance'])) { $conn->executeUpdate('DELETE FROM JoinedChildClass'); $conn->executeUpdate('DELETE FROM JoinedRootClass'); + $conn->executeUpdate('DELETE FROM SingleRootClass'); }