. */ namespace Doctrine\ORM; use Doctrine\Common\EventManager, Doctrine\DBAL\Connection, Doctrine\ORM\Mapping\ClassMetadata, Doctrine\ORM\Mapping\ClassMetadataFactory, Doctrine\ORM\Proxy\ProxyFactory; /** * The EntityManager is the central access point to ORM functionality. * * @since 2.0 * @author Benjamin Eberlei * @author Guilherme Blanco * @author Jonathan Wage * @author Roman Borschel */ class EntityManager { /** * 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; /** * @var ExpressionBuilder The expression builder instance used to generate query expressions. */ private $_expressionBuilder; /** * Whether the EntityManager is closed or not. */ private $_closed = false; /** * The ORM Entity Transaction. * * @var Doctrine\ORM\EntityTransaction */ protected $_transaction; /** * 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; $this->_metadataFactory = new ClassMetadataFactory($this); $this->_metadataFactory->setCacheDriver($this->_config->getMetadataCacheImpl()); $this->_unitOfWork = new UnitOfWork($this); $this->_proxyFactory = new ProxyFactory($this, $config->getProxyDir(), $config->getProxyNamespace(), $config->getAutoGenerateProxyClasses()); $this->_transaction = new EntityTransaction($this); } /** * 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 the ORM Transaction instance. * * @return Doctrine\ORM\EntityTransaction */ public function getTransaction() { return $this->_transaction; } /** * Gets an ExpressionBuilder used for object-oriented construction of query expressions. * * Example: * * [php] * $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))); * * @return ExpressionBuilder */ public function getExpressionBuilder() { if ($this->_expressionBuilder === null) { $this->_expressionBuilder = new Query\Expr; } return $this->_expressionBuilder; } /** * Starts a transaction on the underlying database connection. */ public function beginTransaction() { $this->getTransaction()->begin(); } /** * Commits a transaction on the underlying database connection. */ public function commit() { $this->getTransaction()->commit(); } /** * Performs a rollback on the underlying database connection and closes the * EntityManager as it may now be in a corrupted state. * * @return boolean TRUE on success, FALSE on failure */ public function rollback() { return $this->getTransaction()->rollback(); } /** * Returns the metadata for a class. * * @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, \Doctrine\ORM\Query\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. */ 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 * @return object */ public function find($entityName, $identifier) { return $this->getRepository($entityName)->find($identifier); } /** * Gets a reference to the entity identified by the given type and identifier * without actually loading it. * * If partial objects are allowed, this method will return a partial object that only * has its identifier populated. Otherwise a proxy is returned that automatically * loads itself on first access. * * @return object The entity reference. */ public function getReference($entityName, $identifier) { $class = $this->_metadataFactory->getMetadataFor($entityName); // Check identity map first, if its already in there just return it. if ($entity = $this->_unitOfWork->tryGetById($identifier, $class->rootEntityName)) { return $entity; } 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; } /** * 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."); } /** * Gets the repository for an entity class. * * @param string $entityName The name of the Entity. * @return EntityRepository The repository. */ public function getRepository($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(); } } /** * 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; default: 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; } /** * 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()); } }