123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752 |
- <?php
- /*
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- *
- * This software consists of voluntary contributions made by many individuals
- * and is licensed under the LGPL. For more information, see
- * <http://www.doctrine-project.org>.
- */
-
- namespace Doctrine\ORM;
-
- use Closure, Exception,
- Doctrine\Common\EventManager,
- Doctrine\Common\Persistence\ObjectManager,
- Doctrine\DBAL\Connection,
- Doctrine\DBAL\LockMode,
- Doctrine\ORM\Mapping\ClassMetadata,
- Doctrine\ORM\Mapping\ClassMetadataFactory,
- Doctrine\ORM\Query\ResultSetMapping,
- Doctrine\ORM\Proxy\ProxyFactory;
-
- /**
- * The EntityManager is the central access point to ORM functionality.
- *
- * @since 2.0
- * @author Benjamin Eberlei <kontakt@beberlei.de>
- * @author Guilherme Blanco <guilhermeblanco@hotmail.com>
- * @author Jonathan Wage <jonwage@gmail.com>
- * @author Roman Borschel <roman@code-factory.org>
- */
- class EntityManager implements ObjectManager
- {
- /**
- * The used Configuration.
- *
- * @var \Doctrine\ORM\Configuration
- */
- private $config;
-
- /**
- * The database connection used by the EntityManager.
- *
- * @var \Doctrine\DBAL\Connection
- */
- private $conn;
-
- /**
- * The metadata factory, used to retrieve the ORM metadata of entity classes.
- *
- * @var \Doctrine\ORM\Mapping\ClassMetadataFactory
- */
- private $metadataFactory;
-
- /**
- * The EntityRepository instances.
- *
- * @var array
- */
- private $repositories = array();
-
- /**
- * The UnitOfWork used to coordinate object-level transactions.
- *
- * @var \Doctrine\ORM\UnitOfWork
- */
- private $unitOfWork;
-
- /**
- * The event manager that is the central point of the event system.
- *
- * @var \Doctrine\Common\EventManager
- */
- private $eventManager;
-
- /**
- * The maintained (cached) hydrators. One instance per type.
- *
- * @var array
- */
- private $hydrators = array();
-
- /**
- * The proxy factory used to create dynamic proxies.
- *
- * @var \Doctrine\ORM\Proxy\ProxyFactory
- */
- private $proxyFactory;
-
- /**
- * The expression builder instance used to generate query expressions.
- *
- * @var \Doctrine\ORM\Query\Expr
- */
- private $expressionBuilder;
-
- /**
- * Whether the EntityManager is closed or not.
- *
- * @var bool
- */
- private $closed = false;
-
- /**
- * Creates a new EntityManager that operates on the given database connection
- * and uses the given Configuration and EventManager implementations.
- *
- * @param \Doctrine\DBAL\Connection $conn
- * @param \Doctrine\ORM\Configuration $config
- * @param \Doctrine\Common\EventManager $eventManager
- */
- protected function __construct(Connection $conn, Configuration $config, EventManager $eventManager)
- {
- $this->conn = $conn;
- $this->config = $config;
- $this->eventManager = $eventManager;
-
- $metadataFactoryClassName = $config->getClassMetadataFactoryName();
- $this->metadataFactory = new $metadataFactoryClassName;
- $this->metadataFactory->setEntityManager($this);
- $this->metadataFactory->setCacheDriver($this->config->getMetadataCacheImpl());
-
- $this->unitOfWork = new UnitOfWork($this);
- $this->proxyFactory = new ProxyFactory($this,
- $config->getProxyDir(),
- $config->getProxyNamespace(),
- $config->getAutoGenerateProxyClasses());
- }
-
- /**
- * Gets the database connection object used by the EntityManager.
- *
- * @return \Doctrine\DBAL\Connection
- */
- public function getConnection()
- {
- return $this->conn;
- }
-
- /**
- * Gets the metadata factory used to gather the metadata of classes.
- *
- * @return \Doctrine\ORM\Mapping\ClassMetadataFactory
- */
- public function getMetadataFactory()
- {
- return $this->metadataFactory;
- }
-
- /**
- * Gets an ExpressionBuilder used for object-oriented construction of query expressions.
- *
- * Example:
- *
- * <code>
- * $qb = $em->createQueryBuilder();
- * $expr = $em->getExpressionBuilder();
- * $qb->select('u')->from('User', 'u')
- * ->where($expr->orX($expr->eq('u.id', 1), $expr->eq('u.id', 2)));
- * </code>
- *
- * @return \Doctrine\ORM\Query\Expr
- */
- public function getExpressionBuilder()
- {
- if ($this->expressionBuilder === null) {
- $this->expressionBuilder = new Query\Expr;
- }
- return $this->expressionBuilder;
- }
-
- /**
- * Starts a transaction on the underlying database connection.
- *
- * @deprecated Use {@link getConnection}.beginTransaction().
- */
- public function beginTransaction()
- {
- $this->conn->beginTransaction();
- }
-
- /**
- * Executes a function in a transaction.
- *
- * The function gets passed this EntityManager instance as an (optional) parameter.
- *
- * {@link flush} is invoked prior to transaction commit.
- *
- * If an exception occurs during execution of the function or flushing or transaction commit,
- * the transaction is rolled back, the EntityManager closed and the exception re-thrown.
- *
- * @param Closure $func The function to execute transactionally.
- */
- public function transactional(Closure $func)
- {
- $this->conn->beginTransaction();
-
- try {
- $return = $func($this);
-
- $this->flush();
- $this->conn->commit();
-
- return $return ?: true;
- } catch (Exception $e) {
- $this->close();
- $this->conn->rollback();
-
- throw $e;
- }
- }
-
- /**
- * Commits a transaction on the underlying database connection.
- *
- * @deprecated Use {@link getConnection}.commit().
- */
- public function commit()
- {
- $this->conn->commit();
- }
-
- /**
- * Performs a rollback on the underlying database connection.
- *
- * @deprecated Use {@link getConnection}.rollback().
- */
- public function rollback()
- {
- $this->conn->rollback();
- }
-
- /**
- * Returns the ORM metadata descriptor for a class.
- *
- * The class name must be the fully-qualified class name without a leading backslash
- * (as it is returned by get_class($obj)) or an aliased class name.
- *
- * Examples:
- * MyProject\Domain\User
- * sales:PriceRequest
- *
- * @return \Doctrine\ORM\Mapping\ClassMetadata
- * @internal Performance-sensitive method.
- */
- public function getClassMetadata($className)
- {
- return $this->metadataFactory->getMetadataFor($className);
- }
-
- /**
- * Creates a new Query object.
- *
- * @param string The DQL string.
- * @return \Doctrine\ORM\Query
- */
- public function createQuery($dql = "")
- {
- $query = new Query($this);
- if ( ! empty($dql)) {
- $query->setDql($dql);
- }
- return $query;
- }
-
- /**
- * Creates a Query from a named query.
- *
- * @param string $name
- * @return \Doctrine\ORM\Query
- */
- public function createNamedQuery($name)
- {
- return $this->createQuery($this->config->getNamedQuery($name));
- }
-
- /**
- * Creates a native SQL query.
- *
- * @param string $sql
- * @param ResultSetMapping $rsm The ResultSetMapping to use.
- * @return NativeQuery
- */
- public function createNativeQuery($sql, ResultSetMapping $rsm)
- {
- $query = new NativeQuery($this);
- $query->setSql($sql);
- $query->setResultSetMapping($rsm);
- return $query;
- }
-
- /**
- * Creates a NativeQuery from a named native query.
- *
- * @param string $name
- * @return \Doctrine\ORM\NativeQuery
- */
- public function createNamedNativeQuery($name)
- {
- list($sql, $rsm) = $this->config->getNamedNativeQuery($name);
- return $this->createNativeQuery($sql, $rsm);
- }
-
- /**
- * Create a QueryBuilder instance
- *
- * @return QueryBuilder $qb
- */
- public function createQueryBuilder()
- {
- return new QueryBuilder($this);
- }
-
- /**
- * Flushes all changes to objects that have been queued up to now to the database.
- * This effectively synchronizes the in-memory state of managed objects with the
- * database.
- *
- * @throws \Doctrine\ORM\OptimisticLockException If a version check on an entity that
- * makes use of optimistic locking fails.
- */
- public function flush()
- {
- $this->errorIfClosed();
- $this->unitOfWork->commit();
- }
-
- /**
- * Finds an Entity by its identifier.
- *
- * This is just a convenient shortcut for getRepository($entityName)->find($id).
- *
- * @param string $entityName
- * @param mixed $identifier
- * @param int $lockMode
- * @param int $lockVersion
- * @return object
- */
- public function find($entityName, $identifier, $lockMode = LockMode::NONE, $lockVersion = null)
- {
- return $this->getRepository($entityName)->find($identifier, $lockMode, $lockVersion);
- }
-
- /**
- * Gets a reference to the entity identified by the given type and identifier
- * without actually loading it, if the entity is not yet loaded.
- *
- * @param string $entityName The name of the entity type.
- * @param mixed $identifier The entity identifier.
- * @return object The entity reference.
- */
- public function getReference($entityName, $identifier)
- {
- $class = $this->metadataFactory->getMetadataFor(ltrim($entityName, '\\'));
-
- // Check identity map first, if its already in there just return it.
- if ($entity = $this->unitOfWork->tryGetById($identifier, $class->rootEntityName)) {
- return ($entity instanceof $class->name) ? $entity : null;
- }
- if ($class->subClasses) {
- $entity = $this->find($entityName, $identifier);
- } else {
- if ( ! is_array($identifier)) {
- $identifier = array($class->identifier[0] => $identifier);
- }
- $entity = $this->proxyFactory->getProxy($class->name, $identifier);
- $this->unitOfWork->registerManaged($entity, $identifier, array());
- }
-
- return $entity;
- }
-
- /**
- * Gets a partial reference to the entity identified by the given type and identifier
- * without actually loading it, if the entity is not yet loaded.
- *
- * The returned reference may be a partial object if the entity is not yet loaded/managed.
- * If it is a partial object it will not initialize the rest of the entity state on access.
- * Thus you can only ever safely access the identifier of an entity obtained through
- * this method.
- *
- * The use-cases for partial references involve maintaining bidirectional associations
- * without loading one side of the association or to update an entity without loading it.
- * Note, however, that in the latter case the original (persistent) entity data will
- * never be visible to the application (especially not event listeners) as it will
- * never be loaded in the first place.
- *
- * @param string $entityName The name of the entity type.
- * @param mixed $identifier The entity identifier.
- * @return object The (partial) entity reference.
- */
- public function getPartialReference($entityName, $identifier)
- {
- $class = $this->metadataFactory->getMetadataFor(ltrim($entityName, '\\'));
-
- // Check identity map first, if its already in there just return it.
- if ($entity = $this->unitOfWork->tryGetById($identifier, $class->rootEntityName)) {
- return ($entity instanceof $class->name) ? $entity : null;
- }
- if ( ! is_array($identifier)) {
- $identifier = array($class->identifier[0] => $identifier);
- }
-
- $entity = $class->newInstance();
- $class->setIdentifierValues($entity, $identifier);
- $this->unitOfWork->registerManaged($entity, $identifier, array());
- $this->unitOfWork->markReadOnly($entity);
-
- return $entity;
- }
-
- /**
- * Clears the EntityManager. All entities that are currently managed
- * by this EntityManager become detached.
- *
- * @param string $entityName
- */
- public function clear($entityName = null)
- {
- if ($entityName === null) {
- $this->unitOfWork->clear();
- } else {
- //TODO
- throw new ORMException("EntityManager#clear(\$entityName) not yet implemented.");
- }
- }
-
- /**
- * Closes the EntityManager. All entities that are currently managed
- * by this EntityManager become detached. The EntityManager may no longer
- * be used after it is closed.
- */
- public function close()
- {
- $this->clear();
- $this->closed = true;
- }
-
- /**
- * Tells the EntityManager to make an instance managed and persistent.
- *
- * The entity will be entered into the database at or before transaction
- * commit or as a result of the flush operation.
- *
- * NOTE: The persist operation always considers entities that are not yet known to
- * this EntityManager as NEW. Do not pass detached entities to the persist operation.
- *
- * @param object $object The instance to make managed and persistent.
- */
- public function persist($entity)
- {
- if ( ! is_object($entity)) {
- throw new \InvalidArgumentException(gettype($entity));
- }
- $this->errorIfClosed();
- $this->unitOfWork->persist($entity);
- }
-
- /**
- * Removes an entity instance.
- *
- * A removed entity will be removed from the database at or before transaction commit
- * or as a result of the flush operation.
- *
- * @param object $entity The entity instance to remove.
- */
- public function remove($entity)
- {
- if ( ! is_object($entity)) {
- throw new \InvalidArgumentException(gettype($entity));
- }
- $this->errorIfClosed();
- $this->unitOfWork->remove($entity);
- }
-
- /**
- * Refreshes the persistent state of an entity from the database,
- * overriding any local changes that have not yet been persisted.
- *
- * @param object $entity The entity to refresh.
- */
- public function refresh($entity)
- {
- if ( ! is_object($entity)) {
- throw new \InvalidArgumentException(gettype($entity));
- }
- $this->errorIfClosed();
- $this->unitOfWork->refresh($entity);
- }
-
- /**
- * Detaches an entity from the EntityManager, causing a managed entity to
- * become detached. Unflushed changes made to the entity if any
- * (including removal of the entity), will not be synchronized to the database.
- * Entities which previously referenced the detached entity will continue to
- * reference it.
- *
- * @param object $entity The entity to detach.
- */
- public function detach($entity)
- {
- if ( ! is_object($entity)) {
- throw new \InvalidArgumentException(gettype($entity));
- }
- $this->unitOfWork->detach($entity);
- }
-
- /**
- * Merges the state of a detached entity into the persistence context
- * of this EntityManager and returns the managed copy of the entity.
- * The entity passed to merge will not become associated/managed with this EntityManager.
- *
- * @param object $entity The detached entity to merge into the persistence context.
- * @return object The managed copy of the entity.
- */
- public function merge($entity)
- {
- if ( ! is_object($entity)) {
- throw new \InvalidArgumentException(gettype($entity));
- }
- $this->errorIfClosed();
- return $this->unitOfWork->merge($entity);
- }
-
- /**
- * Creates a copy of the given entity. Can create a shallow or a deep copy.
- *
- * @param object $entity The entity to copy.
- * @return object The new entity.
- * @todo Implementation need. This is necessary since $e2 = clone $e1; throws an E_FATAL when access anything on $e:
- * Fatal error: Maximum function nesting level of '100' reached, aborting!
- */
- public function copy($entity, $deep = false)
- {
- throw new \BadMethodCallException("Not implemented.");
- }
-
- /**
- * Acquire a lock on the given entity.
- *
- * @param object $entity
- * @param int $lockMode
- * @param int $lockVersion
- * @throws OptimisticLockException
- * @throws PessimisticLockException
- */
- public function lock($entity, $lockMode, $lockVersion = null)
- {
- $this->unitOfWork->lock($entity, $lockMode, $lockVersion);
- }
-
- /**
- * Gets the repository for an entity class.
- *
- * @param string $entityName The name of the entity.
- * @return EntityRepository The repository class.
- */
- public function getRepository($entityName)
- {
- $entityName = ltrim($entityName, '\\');
- if (isset($this->repositories[$entityName])) {
- return $this->repositories[$entityName];
- }
-
- $metadata = $this->getClassMetadata($entityName);
- $customRepositoryClassName = $metadata->customRepositoryClassName;
-
- if ($customRepositoryClassName !== null) {
- $repository = new $customRepositoryClassName($this, $metadata);
- } else {
- $repository = new EntityRepository($this, $metadata);
- }
-
- $this->repositories[$entityName] = $repository;
-
- return $repository;
- }
-
- /**
- * Determines whether an entity instance is managed in this EntityManager.
- *
- * @param object $entity
- * @return boolean TRUE if this EntityManager currently manages the given entity, FALSE otherwise.
- */
- public function contains($entity)
- {
- return $this->unitOfWork->isScheduledForInsert($entity) ||
- $this->unitOfWork->isInIdentityMap($entity) &&
- ! $this->unitOfWork->isScheduledForDelete($entity);
- }
-
- /**
- * Gets the EventManager used by the EntityManager.
- *
- * @return \Doctrine\Common\EventManager
- */
- public function getEventManager()
- {
- return $this->eventManager;
- }
-
- /**
- * Gets the Configuration used by the EntityManager.
- *
- * @return \Doctrine\ORM\Configuration
- */
- public function getConfiguration()
- {
- return $this->config;
- }
-
- /**
- * Throws an exception if the EntityManager is closed or currently not active.
- *
- * @throws ORMException If the EntityManager is closed.
- */
- private function errorIfClosed()
- {
- if ($this->closed) {
- throw ORMException::entityManagerClosed();
- }
- }
-
- /**
- * Check if the Entity manager is open or closed.
- *
- * @return bool
- */
- public function isOpen()
- {
- return (!$this->closed);
- }
-
- /**
- * Gets the UnitOfWork used by the EntityManager to coordinate operations.
- *
- * @return \Doctrine\ORM\UnitOfWork
- */
- public function getUnitOfWork()
- {
- return $this->unitOfWork;
- }
-
- /**
- * Gets a hydrator for the given hydration mode.
- *
- * This method caches the hydrator instances which is used for all queries that don't
- * selectively iterate over the result.
- *
- * @param int $hydrationMode
- * @return \Doctrine\ORM\Internal\Hydration\AbstractHydrator
- */
- public function getHydrator($hydrationMode)
- {
- if ( ! isset($this->hydrators[$hydrationMode])) {
- $this->hydrators[$hydrationMode] = $this->newHydrator($hydrationMode);
- }
-
- return $this->hydrators[$hydrationMode];
- }
-
- /**
- * Create a new instance for the given hydration mode.
- *
- * @param int $hydrationMode
- * @return \Doctrine\ORM\Internal\Hydration\AbstractHydrator
- */
- public function newHydrator($hydrationMode)
- {
- switch ($hydrationMode) {
- case Query::HYDRATE_OBJECT:
- $hydrator = new Internal\Hydration\ObjectHydrator($this);
- break;
- case Query::HYDRATE_ARRAY:
- $hydrator = new Internal\Hydration\ArrayHydrator($this);
- break;
- case Query::HYDRATE_SCALAR:
- $hydrator = new Internal\Hydration\ScalarHydrator($this);
- break;
- case Query::HYDRATE_SINGLE_SCALAR:
- $hydrator = new Internal\Hydration\SingleScalarHydrator($this);
- break;
- case Query::HYDRATE_SIMPLEOBJECT:
- $hydrator = new Internal\Hydration\SimpleObjectHydrator($this);
- break;
- default:
- if ($class = $this->config->getCustomHydrationMode($hydrationMode)) {
- $hydrator = new $class($this);
- break;
- }
- throw ORMException::invalidHydrationMode($hydrationMode);
- }
-
- return $hydrator;
- }
-
- /**
- * Gets the proxy factory used by the EntityManager to create entity proxies.
- *
- * @return ProxyFactory
- */
- public function getProxyFactory()
- {
- return $this->proxyFactory;
- }
-
- /**
- * {@inheritDoc}
- */
- public function initializeObject($entity)
- {
- $this->unitOfWork->initializeObject($entity);
- }
-
- /**
- * Factory method to create EntityManager instances.
- *
- * @param mixed $conn An array with the connection parameters or an existing
- * Connection instance.
- * @param Configuration $config The Configuration instance to use.
- * @param EventManager $eventManager The EventManager instance to use.
- * @return EntityManager The created EntityManager.
- */
- public static function create($conn, Configuration $config, EventManager $eventManager = null)
- {
- if (!$config->getMetadataDriverImpl()) {
- throw ORMException::missingMappingDriverImpl();
- }
-
- if (is_array($conn)) {
- $conn = \Doctrine\DBAL\DriverManager::getConnection($conn, $config, ($eventManager ?: new EventManager()));
- } else if ($conn instanceof Connection) {
- if ($eventManager !== null && $conn->getEventManager() !== $eventManager) {
- throw ORMException::mismatchedEventManager();
- }
- } else {
- throw new \InvalidArgumentException("Invalid argument: " . $conn);
- }
-
- return new EntityManager($conn, $config, $conn->getEventManager());
- }
- }
|