. */ namespace Doctrine\ORM\Persisters; use Doctrine\Common\Collections\Criteria; use Doctrine\ORM\PersistentCollection; use Doctrine\ORM\UnitOfWork; /** * Persister for one-to-many collections. * * @author Roman Borschel * @author Guilherme Blanco * @author Alexander * @since 2.0 */ class OneToManyPersister extends AbstractCollectionPersister { /** * {@inheritdoc} */ public function get(PersistentCollection $coll, $index) { $mapping = $coll->getMapping(); $uow = $this->em->getUnitOfWork(); $persister = $uow->getEntityPersister($mapping['targetEntity']); if (!isset($mapping['indexBy'])) { throw new \BadMethodCallException("Selecting a collection by index is only supported on indexed collections."); } return $persister->load(array($mapping['mappedBy'] => $coll->getOwner(), $mapping['indexBy'] => $index), null, null, array(), null, 1); } /** * Generates the SQL UPDATE that updates a particular row's foreign * key to null. * * @param \Doctrine\ORM\PersistentCollection $coll * * @return string * * @override */ protected function getDeleteRowSQL(PersistentCollection $coll) { $mapping = $coll->getMapping(); $class = $this->em->getClassMetadata($mapping['targetEntity']); $tableName = $this->quoteStrategy->getTableName($class, $this->platform); $idColumns = $class->getIdentifierColumnNames(); return 'DELETE FROM ' . $tableName . ' WHERE ' . implode('= ? AND ', $idColumns) . ' = ?'; } /** * {@inheritdoc} */ protected function getDeleteRowSQLParameters(PersistentCollection $coll, $element) { return array_values($this->uow->getEntityIdentifier($element)); } /** * {@inheritdoc} * * @throws \BadMethodCallException Not used for OneToManyPersister. */ protected function getInsertRowSQL(PersistentCollection $coll) { throw new \BadMethodCallException("Insert Row SQL is not used for OneToManyPersister"); } /** * {@inheritdoc} * * @throws \BadMethodCallException Not used for OneToManyPersister. */ protected function getInsertRowSQLParameters(PersistentCollection $coll, $element) { throw new \BadMethodCallException("Insert Row SQL is not used for OneToManyPersister"); } /** * {@inheritdoc} * * @throws \BadMethodCallException Not used for OneToManyPersister. */ protected function getUpdateRowSQL(PersistentCollection $coll) { throw new \BadMethodCallException("Update Row SQL is not used for OneToManyPersister"); } /** * {@inheritdoc} * * @throws \BadMethodCallException Not used for OneToManyPersister. */ protected function getDeleteSQL(PersistentCollection $coll) { throw new \BadMethodCallException("Delete Row SQL is not used for OneToManyPersister"); } /** * {@inheritdoc} * * @throws \BadMethodCallException Not used for OneToManyPersister. */ protected function getDeleteSQLParameters(PersistentCollection $coll) { throw new \BadMethodCallException("Delete Row SQL is not used for OneToManyPersister"); } /** * {@inheritdoc} */ public function count(PersistentCollection $coll) { $mapping = $coll->getMapping(); $uow = $this->em->getUnitOfWork(); $persister = $uow->getEntityPersister($mapping['targetEntity']); // only works with single id identifier entities. Will throw an // exception in Entity Persisters if that is not the case for the // 'mappedBy' field. $criteria = new Criteria(Criteria::expr()->eq($mapping['mappedBy'], $coll->getOwner())); return $persister->count($criteria); } /** * {@inheritdoc} */ public function slice(PersistentCollection $coll, $offset, $length = null) { $mapping = $coll->getMapping(); $uow = $this->em->getUnitOfWork(); $persister = $uow->getEntityPersister($mapping['targetEntity']); return $persister->getOneToManyCollection($mapping, $coll->getOwner(), $offset, $length); } /** * {@inheritdoc} */ public function containsKey(PersistentCollection $coll, $key) { $mapping = $coll->getMapping(); $uow = $this->em->getUnitOfWork(); $persister = $uow->getEntityPersister($mapping['targetEntity']); // only works with single id identifier entities. Will throw an // exception in Entity Persisters if that is not the case for the // 'mappedBy' field. $criteria = new Criteria(); $criteria->andWhere(Criteria::expr()->eq($mapping['mappedBy'], $coll->getOwner())); $criteria->andWhere(Criteria::expr()->eq($mapping['indexBy'], $key)); return (bool) $persister->count($criteria); } /** * {@inheritdoc} */ public function contains(PersistentCollection $coll, $element) { $mapping = $coll->getMapping(); $uow = $this->em->getUnitOfWork(); // shortcut for new entities $entityState = $uow->getEntityState($element, UnitOfWork::STATE_NEW); if ($entityState === UnitOfWork::STATE_NEW) { return false; } // Entity is scheduled for inclusion if ($entityState === UnitOfWork::STATE_MANAGED && $uow->isScheduledForInsert($element)) { return false; } $persister = $uow->getEntityPersister($mapping['targetEntity']); // only works with single id identifier entities. Will throw an // exception in Entity Persisters if that is not the case for the // 'mappedBy' field. $criteria = new Criteria(Criteria::expr()->eq($mapping['mappedBy'], $coll->getOwner())); return $persister->exists($element, $criteria); } /** * {@inheritdoc} */ public function removeElement(PersistentCollection $coll, $element) { $uow = $this->em->getUnitOfWork(); // shortcut for new entities $entityState = $uow->getEntityState($element, UnitOfWork::STATE_NEW); if ($entityState === UnitOfWork::STATE_NEW) { return false; } // If Entity is scheduled for inclusion, it is not in this collection. // We can assure that because it would have return true before on array check if ($entityState === UnitOfWork::STATE_MANAGED && $uow->isScheduledForInsert($element)) { return false; } $mapping = $coll->getMapping(); $persister = $uow->getEntityPersister($mapping['targetEntity']); return $persister->delete($element); } }