. */ #namespace Doctrine::ORM; #use Doctrine::Common::Configuration; #use Doctrine::Common::EventManager; #use Doctrine::Common::NullObject; #use Doctrine::DBAL::Connections::Connection; #use Doctrine::ORM::Exceptions::EntityManagerException; #use Doctrine::ORM::Internal::UnitOfWork; #use Doctrine::ORM::Mapping::ClassMetadata; /** * The EntityManager is the central access point to ORM functionality. * * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @link www.phpdoctrine.org * @since 2.0 * @version $Revision$ * @author Roman Borschel */ class Doctrine_EntityManager { /** * IMMEDIATE: Flush occurs automatically after each operation that issues database * queries. No operations are queued. */ const FLUSHMODE_IMMEDIATE = 'immediate'; /** * AUTO: Flush occurs automatically in the following situations: * - Before any query executions (to prevent getting stale data) * - On EntityManager#commit() */ const FLUSHMODE_AUTO = 'auto'; /** * COMMIT: Flush occurs automatically only on EntityManager#commit(). */ const FLUSHMODE_COMMIT = 'commit'; /** * MANUAL: Flush occurs never automatically. The only way to flush is * through EntityManager#flush(). */ const FLUSHMODE_MANUAL = 'manual'; /** * The currently active EntityManager. Only one EntityManager can be active * at any time. * * @var Doctrine::ORM::EntityManager */ private static $_activeEm; /** * The unique name of the EntityManager. The name is used to bind entity classes * to certain EntityManagers. * * @var string */ private $_name; /** * The used Configuration. * * @var Configuration */ private $_config; /** * The database connection used by the EntityManager. * * @var Doctrine_Connection */ private $_conn; /** * The metadata factory, used to retrieve the metadata of entity classes. * * @var Doctrine::ORM::Mapping::ClassMetadataFactory */ private $_metadataFactory; /** * The EntityPersister instances. * * @var array */ private $_persisters = array(); /** * The EntityRepository instances. * * @var array */ private $_repositories = array(); /** * The currently used flush mode. Defaults to 'commit'. * * @var string */ private $_flushMode = 'commit'; /** * The unit of work used to coordinate object-level transactions. * * @var UnitOfWork */ private $_unitOfWork; /** * The event manager that is the central point of the event system. * * @var EventManager */ private $_eventManager; /** * Container that is used temporarily during hydration. * * @var array */ private $_tmpEntityData = array(); private $_closed = false; /** * Creates a new EntityManager that operates on the given database connection. * * @param Doctrine_Connection $conn * @param string $name */ protected function __construct(Doctrine_Connection $conn, $name, Doctrine_Configuration $config, Doctrine_EventManager $eventManager) { $this->_conn = $conn; $this->_name = $name; $this->_config = $config; $this->_eventManager = $eventManager; $this->_metadataFactory = new Doctrine_ClassMetadata_Factory( $this, new Doctrine_ClassMetadata_CodeDriver()); $this->_unitOfWork = new Doctrine_Connection_UnitOfWork($this); $this->_nullObject = Doctrine_ORM_Internal_Null::$INSTANCE; } /** * Gets the database connection object used by the EntityManager. * * @return Doctrine_Connection */ public function getConnection() { return $this->_conn; } /** * Gets the name of the EntityManager. * * @return string The name of the EntityManager. */ public function getName() { return $this->_name; } /** * Gets the metadata for a class. Alias for getClassMetadata(). * * @return Doctrine_Metadata */ public function getMetadata($className) { return $this->getClassMetadata($className); } /** * Starts a database transaction. */ public function beginTransaction() { return $this->_conn->beginTransaction(); } /** * Commits a running database transaction. * This causes a flush() of the EntityManager if the flush mode is set to * AUTO or COMMIT. * * @return unknown */ public function commit() { if ($this->_flushMode == self::FLUSHMODE_AUTO || $this->_flushMode == self::FLUSHMODE_COMMIT) { $this->flush(); } return $this->_conn->commitTransaction(); } /** * Returns the metadata for a class. * * @return Doctrine_Metadata */ public function getClassMetadata($className) { return $this->_metadataFactory->getMetadataFor($className); } /** * Sets the driver that is used to obtain metadata mapping information * about Entities. * * @param $driver The driver to use. */ public function setClassMetadataDriver($driver) { $this->_metadataFactory->setDriver($driver); } /** * Creates a new Doctrine_Query object that operates on this connection. * * @param string The DQL string. * @return Doctrine::ORM::Query * @todo package:orm */ public function createQuery($dql = "") { $query = new Doctrine_Query($this); if ( ! empty($dql)) { $query->setDql($dql); } return $query; } /** * Gets the EntityPersister for an Entity. * * This is usually not of interest for users, mainly for internal use. * * @param string $entityName The name of the Entity. * @return Doctrine::ORM::Internal::EntityPersister */ public function getEntityPersister($entityName) { if ( ! isset($this->_persisters[$entityName])) { $class = $this->getClassMetadata($entityName); if ($class->getInheritanceType() == Doctrine::INHERITANCE_TYPE_JOINED) { $persister = new Doctrine_EntityPersister_JoinedSubclass($this, $class); } else { $persister = new Doctrine_EntityPersister_Standard($this, $class); } $this->_persisters[$entityName] = $persister; } return $this->_persisters[$entityName]; } /** * Detaches an entity from the manager. It's lifecycle is no longer managed. * * @param Doctrine_Entity $entity * @return unknown */ public function detach(Doctrine_Entity $entity) { return $this->_unitOfWork->removeFromIdentityMap($entity); } /** * Creates a query with the specified name. * * @todo Implementation. * @throws SomeException If there is no query registered with the given name. */ public function createNamedQuery($name) { //... } /** * @todo Implementation. */ public function createNativeQuery($sql = "") { //... } /** * @todo Implementation. */ public function createNamedNativeQuery($name) { //... } /** * @todo Implementation. */ public function createCriteria() { //... } /** * Flushes all changes to objects that have been queued up to now to the database. */ public function flush() { $this->_errorIfNotActiveOrClosed(); $this->_unitOfWork->commit(); } /** * Finds an Entity by its identifier. * This is just a convenient shortcut for getRepository()->find(). * * @param string $entityName * @param mixed $identifier * @return Doctrine::ORM::Entity */ public function find($entityName, $identifier) { return $this->getRepository($entityName)->find($identifier); } /** * Sets the flush mode. * * @param string $flushMode */ public function setFlushMode($flushMode) { if ( ! $this->_isFlushMode($flushMode)) { throw Doctrine_ORM_Exceptions_EntityManagerException::invalidFlushMode(); } $this->_flushMode = $flushMode; } private function _isFlushMode($value) { return $value == self::FLUSHMODE_AUTO || $value == self::FLUSHMODE_COMMIT || $value == self::FLUSHMODE_IMMEDIATE || $value == self::FLUSHMODE_MANUAL; } /** * Gets the currently used flush mode. * * @return string */ public function getFlushMode() { return $this->_flushMode; } /** * Clears the persistence context, detaching all entities. * * @return void * @todo package:orm */ public function clear($entityName = null) { if ($entityName === null) { $this->_unitOfWork->detachAll(); } else { //... } } /** * Closes the EntityManager. * */ public function close() { $this->_closed = true; } /** * Gets the result cache driver used by the EntityManager. * * @return Doctrine::ORM::Cache::CacheDriver The cache driver. */ public function getResultCacheDriver() { if ( ! $this->getAttribute(Doctrine::ATTR_RESULT_CACHE)) { throw new Doctrine_Exception('Result Cache driver not initialized.'); } return $this->getAttribute(Doctrine::ATTR_RESULT_CACHE); } /** * getQueryCacheDriver * * @return Doctrine_Cache_Interface */ public function getQueryCacheDriver() { if ( ! $this->getAttribute(Doctrine::ATTR_QUERY_CACHE)) { throw new Doctrine_Exception('Query Cache driver not initialized.'); } return $this->getAttribute(Doctrine::ATTR_QUERY_CACHE); } /** * Saves the given entity, persisting it's state. * * @param Doctrine::ORM::Entity $entity * @return void */ public function save(Doctrine_Entity $entity) { $this->_errorIfNotActiveOrClosed(); $this->_unitOfWork->save($entity); if ($this->_flushMode == self::FLUSHMODE_IMMEDIATE) { $this->flush(); } } /** * Removes the given entity from the persistent store. * * @param Doctrine::ORM::Entity $entity * @return void */ public function delete(Doctrine_Entity $entity) { $this->_errorIfNotActiveOrClosed(); $this->_unitOfWork->delete($entity); if ($this->_flushMode == self::FLUSHMODE_IMMEDIATE) { $this->flush(); } } /** * Refreshes the persistent state of the entity from the database, * overriding any local changes that have not yet been persisted. * * @param Doctrine::ORM::Entity $entity * @return void * @todo FIX Impl */ public function refresh(Doctrine_Entity $entity) { $this->_mergeData($entity, $entity->getRepository()->find( $entity->identifier(), Doctrine_Query::HYDRATE_ARRAY), true); } /** * Creates a copy of the given entity. Can create a shallow or a deep copy. * * @param Doctrine::ORM::Entity $entity The entity to copy. * @return Doctrine::ORM::Entity The new entity. */ public function copy(Doctrine_Entity $entity, $deep = false) { //... } /** * Gets the repository for an Entity. * * @param string $entityName The name of the Entity. * @return Doctrine::ORM::EntityRepository The repository. */ public function getRepository($entityName) { if (isset($this->_repositories[$entityName])) { return $this->_repositories[$entityName]; } $metadata = $this->getClassMetadata($entityName); $customRepositoryClassName = $metadata->getCustomRepositoryClass(); if ($customRepositoryClassName !== null) { $repository = new $customRepositoryClassName($entityName, $metadata); } else { $repository = new Doctrine_EntityRepository($entityName, $metadata); } $this->_repositories[$entityName] = $repository; return $repository; } /** * Creates an entity. Used for reconstitution as well as initial creation. * * @param string $className The name of the entity class. * @param array $data The data for the entity. * @return Doctrine::ORM::Entity */ public function createEntity($className, array $data) { $this->_errorIfNotActiveOrClosed(); $this->_tmpEntityData = $data; $className = $this->_inferCorrectClassName($data, $className); $classMetadata = $this->getClassMetadata($className); if ( ! empty($data)) { $identifierFieldNames = $classMetadata->getIdentifier(); $isNew = false; foreach ($identifierFieldNames as $fieldName) { if ( ! isset($data[$fieldName])) { // id field not found return new entity $isNew = true; break; } $id[] = $data[$fieldName]; } if ($isNew) { $entity = new $className; } else { $idHash = $this->_unitOfWork->getIdentifierHash($id); if ($entity = $this->_unitOfWork->tryGetByIdHash($idHash, $classMetadata->getRootClassName())) { $this->_mergeData($entity, $data); return $entity; } else { $entity = new $className; $this->_unitOfWork->addToIdentityMap($entity); } } } else { $entity = new $className; } return $entity; } /** * Merges the given data into the given entity, optionally overriding * local changes. * * @param Doctrine::ORM::Entity $entity * @param array $data * @param boolean $overrideLocalChanges * @return void */ private function _mergeData(Doctrine_Entity $entity, array $data, $overrideLocalChanges = false) { if ($overrideLocalChanges) { foreach ($data as $field => $value) { $entity->_internalSetField($field, $value); } } else { foreach ($data as $field => $value) { $currentValue = $entity->get($field); if ( ! isset($currentValue) || $entity->_internalGetField($field) === null) { $entity->_internalSetField($field, $value); } } } } /** * Checks if the instance is managed by the EntityManager. * * @param Doctrine::ORM::Entity $entity * @return boolean TRUE if this EntityManager currently manages the given entity * (and has it in the identity map), FALSE otherwise. */ public function contains(Doctrine_Entity $entity) { return $this->_unitOfWork->isInIdentityMap($entity) && ! $this->_unitOfWork->isRegisteredRemoved($entity); } /** * INTERNAL: For internal hydration purposes only. * * Gets the temporarily stored entity data. * * @return array */ public function _getTmpEntityData() { $data = $this->_tmpEntityData; $this->_tmpEntityData = array(); return $data; } /** * Check the dataset for a discriminator column to determine the correct * class to instantiate. If no discriminator column is found, the given * classname will be returned. * * @param array $data * @param string $className * @return string The name of the class to instantiate. */ private function _inferCorrectClassName(array $data, $className) { $class = $this->getClassMetadata($className); $discCol = $class->getInheritanceOption('discriminatorColumn'); if ( ! $discCol) { return $className; } $discMap = $class->getInheritanceOption('discriminatorMap'); if (isset($data[$discCol], $discMap[$data[$discCol]])) { return $discMap[$data[$discCol]]; } else { return $className; } } /** * 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::Common::Configuration */ public function getConfiguration() { return $this->_config; } private function _errorIfNotActiveOrClosed() { if ( ! $this->isActive() || $this->_closed) { throw Doctrine_EntityManagerException::notActiveOrClosed($this->_name); } } /** * Gets the UnitOfWork used by the EntityManager to coordinate operations. * * @return Doctrine::ORM::UnitOfWork */ public function getUnitOfWork() { return $this->_unitOfWork; } /** * Checks whether this EntityManager is the currently active one. * * @return boolean */ public function isActive() { return self::$_activeEm === $this; } /** * Makes this EntityManager the currently active one. * * @return void */ public function activate() { self::$_activeEm = $this; } /** * Factory method to create EntityManager instances. * A newly created EntityManager is immediately activated, making it the * currently active EntityManager. * * @param mixed $conn An array with the connection parameters or an existing * Doctrine::DBAL::Connection instance. * @param string $name * @param Doctrine::Common::Configuration $config The Configuration instance to use. * @param Doctrine::Common::EventManager $eventManager The EventManager instance to use. * @return Doctrine::ORM::EntityManager The created EntityManager. */ public static function create($conn, $name, Doctrine_Configuration $config = null, Doctrine_EventManager $eventManager = null) { if (is_array($conn)) { $connFactory = new Doctrine_ConnectionFactory(); $conn = $connFactory->createConnection($conn, $config, $eventManager); } else if ( ! $conn instanceof Doctrine_Connection) { throw new Doctrine_Exception("Invalid parameter '$conn'."); } if (is_null($config)) { $config = new Doctrine_Configuration(); } if (is_null($eventManager)) { $eventManager = new Doctrine_EventManager(); } $em = new Doctrine_EntityManager($conn, $name, $config, $eventManager); $em->activate(); return $em; } /** * Gets the currently active EntityManager. * * @return Doctrine::ORM::EntityManager */ public static function getActiveEntityManager() { return self::$_activeEm; } } ?>