123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983 |
- <?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 MIT license. For more information, see
- * <http://www.doctrine-project.org>.
- */
- namespace Doctrine\ORM;
- use BadMethodCallException;
- use Doctrine\Common\EventManager;
- use Doctrine\Common\Util\ClassUtils;
- use Doctrine\DBAL\Connection;
- use Doctrine\DBAL\DriverManager;
- use Doctrine\DBAL\LockMode;
- use Doctrine\ORM\Mapping\ClassMetadata;
- use Doctrine\ORM\Mapping\ClassMetadataFactory;
- use Doctrine\ORM\Proxy\ProxyFactory;
- use Doctrine\ORM\Query\Expr;
- use Doctrine\ORM\Query\FilterCollection;
- use Doctrine\ORM\Query\ResultSetMapping;
- use Doctrine\ORM\Repository\RepositoryFactory;
- use Doctrine\Persistence\Mapping\MappingException;
- use Doctrine\Persistence\ObjectRepository;
- use InvalidArgumentException;
- use Throwable;
- use function array_keys;
- use function call_user_func;
- use function get_class;
- use function gettype;
- use function is_array;
- use function is_callable;
- use function is_object;
- use function is_string;
- use function ltrim;
- use function sprintf;
- use function trigger_error;
- use const E_USER_DEPRECATED;
- /**
- * The EntityManager is the central access point to ORM functionality.
- *
- * It is a facade to all different ORM subsystems such as UnitOfWork,
- * Query Language and Repository API. Instantiation is done through
- * the static create() method. The quickest way to obtain a fully
- * configured EntityManager is:
- *
- * use Doctrine\ORM\Tools\Setup;
- * use Doctrine\ORM\EntityManager;
- *
- * $paths = array('/path/to/entity/mapping/files');
- *
- * $config = Setup::createAnnotationMetadataConfiguration($paths);
- * $dbParams = array('driver' => 'pdo_sqlite', 'memory' => true);
- * $entityManager = EntityManager::create($dbParams, $config);
- *
- * For more information see
- * {@link http://docs.doctrine-project.org/projects/doctrine-orm/en/latest/reference/configuration.html}
- *
- * You should never attempt to inherit from the EntityManager: Inheritance
- * is not a valid extension point for the EntityManager. Instead you
- * should take a look at the {@see \Doctrine\ORM\Decorator\EntityManagerDecorator}
- * and wrap your entity manager in a decorator.
- */
- /* final */class EntityManager implements EntityManagerInterface
- {
- /**
- * The used Configuration.
- *
- * @var Configuration
- */
- private $config;
- /**
- * The database connection used by the EntityManager.
- *
- * @var Connection
- */
- private $conn;
- /**
- * The metadata factory, used to retrieve the ORM metadata of entity classes.
- *
- * @var ClassMetadataFactory
- */
- private $metadataFactory;
- /**
- * The UnitOfWork 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;
- /**
- * The proxy factory used to create dynamic proxies.
- *
- * @var ProxyFactory
- */
- private $proxyFactory;
- /**
- * The repository factory used to create dynamic repositories.
- *
- * @var RepositoryFactory
- */
- private $repositoryFactory;
- /**
- * The expression builder instance used to generate query expressions.
- *
- * @var Expr
- */
- private $expressionBuilder;
- /**
- * Whether the EntityManager is closed or not.
- *
- * @var bool
- */
- private $closed = false;
- /**
- * Collection of query filters.
- *
- * @var FilterCollection
- */
- private $filterCollection;
- /** @var Cache The second level cache regions API. */
- private $cache;
- /**
- * Creates a new EntityManager that operates on the given database connection
- * and uses the given Configuration and EventManager implementations.
- */
- 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->repositoryFactory = $config->getRepositoryFactory();
- $this->unitOfWork = new UnitOfWork($this);
- $this->proxyFactory = new ProxyFactory(
- $this,
- $config->getProxyDir(),
- $config->getProxyNamespace(),
- $config->getAutoGenerateProxyClasses()
- );
- if ($config->isSecondLevelCacheEnabled()) {
- $cacheConfig = $config->getSecondLevelCacheConfiguration();
- $cacheFactory = $cacheConfig->getCacheFactory();
- $this->cache = $cacheFactory->createCache($this);
- }
- }
- /**
- * {@inheritDoc}
- */
- public function getConnection()
- {
- return $this->conn;
- }
- /**
- * Gets the metadata factory used to gather the metadata of classes.
- *
- * @return ClassMetadataFactory
- */
- public function getMetadataFactory()
- {
- return $this->metadataFactory;
- }
- /**
- * {@inheritDoc}
- */
- public function getExpressionBuilder()
- {
- if ($this->expressionBuilder === null) {
- $this->expressionBuilder = new Query\Expr();
- }
- return $this->expressionBuilder;
- }
- /**
- * {@inheritDoc}
- */
- public function beginTransaction()
- {
- $this->conn->beginTransaction();
- }
- /**
- * {@inheritDoc}
- */
- public function getCache()
- {
- return $this->cache;
- }
- /**
- * {@inheritDoc}
- */
- public function transactional($func)
- {
- if (! is_callable($func)) {
- throw new InvalidArgumentException('Expected argument of type "callable", got "' . gettype($func) . '"');
- }
- $this->conn->beginTransaction();
- try {
- $return = call_user_func($func, $this);
- $this->flush();
- $this->conn->commit();
- return $return ?: true;
- } catch (Throwable $e) {
- $this->close();
- $this->conn->rollBack();
- throw $e;
- }
- }
- /**
- * {@inheritDoc}
- */
- public function commit()
- {
- $this->conn->commit();
- }
- /**
- * {@inheritDoc}
- */
- 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
- *
- * Internal note: Performance-sensitive method.
- *
- * @param string $className
- *
- * @return ClassMetadata
- */
- public function getClassMetadata($className)
- {
- return $this->metadataFactory->getMetadataFor($className);
- }
- /**
- * {@inheritDoc}
- */
- public function createQuery($dql = '')
- {
- $query = new Query($this);
- if (! empty($dql)) {
- $query->setDQL($dql);
- }
- return $query;
- }
- /**
- * {@inheritDoc}
- */
- public function createNamedQuery($name)
- {
- return $this->createQuery($this->config->getNamedQuery($name));
- }
- /**
- * {@inheritDoc}
- */
- public function createNativeQuery($sql, ResultSetMapping $rsm)
- {
- $query = new NativeQuery($this);
- $query->setSQL($sql);
- $query->setResultSetMapping($rsm);
- return $query;
- }
- /**
- * {@inheritDoc}
- */
- public function createNamedNativeQuery($name)
- {
- [$sql, $rsm] = $this->config->getNamedNativeQuery($name);
- return $this->createNativeQuery($sql, $rsm);
- }
- /**
- * {@inheritDoc}
- */
- 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.
- *
- * If an entity is explicitly passed to this method only this entity and
- * the cascade-persist semantics + scheduled inserts/removals are synchronized.
- *
- * @param object|mixed[]|null $entity
- *
- * @return void
- *
- * @throws OptimisticLockException If a version check on an entity that
- * makes use of optimistic locking fails.
- * @throws ORMException
- */
- public function flush($entity = null)
- {
- if ($entity !== null) {
- @trigger_error(
- 'Calling ' . __METHOD__ . '() with any arguments to flush specific entities is deprecated and will not be supported in Doctrine ORM 3.0.',
- E_USER_DEPRECATED
- );
- }
- $this->errorIfClosed();
- $this->unitOfWork->commit($entity);
- }
- /**
- * Finds an Entity by its identifier.
- *
- * @param string $className The class name of the entity to find.
- * @param mixed $id The identity of the entity to find.
- * @param int|null $lockMode One of the \Doctrine\DBAL\LockMode::* constants
- * or NULL if no specific lock mode should be used
- * during the search.
- * @param int|null $lockVersion The version of the entity to find when using
- * optimistic locking.
- *
- * @return object|null The entity instance or NULL if the entity can not be found.
- *
- * @throws OptimisticLockException
- * @throws ORMInvalidArgumentException
- * @throws TransactionRequiredException
- * @throws ORMException
- *
- * @template T
- * @psalm-param class-string<T> $className
- * @psalm-return ?T
- */
- public function find($className, $id, $lockMode = null, $lockVersion = null)
- {
- $class = $this->metadataFactory->getMetadataFor(ltrim($className, '\\'));
- if ($lockMode !== null) {
- $this->checkLockRequirements($lockMode, $class);
- }
- if (! is_array($id)) {
- if ($class->isIdentifierComposite) {
- throw ORMInvalidArgumentException::invalidCompositeIdentifier();
- }
- $id = [$class->identifier[0] => $id];
- }
- foreach ($id as $i => $value) {
- if (is_object($value) && $this->metadataFactory->hasMetadataFor(ClassUtils::getClass($value))) {
- $id[$i] = $this->unitOfWork->getSingleIdentifierValue($value);
- if ($id[$i] === null) {
- throw ORMInvalidArgumentException::invalidIdentifierBindingEntity();
- }
- }
- }
- $sortedId = [];
- foreach ($class->identifier as $identifier) {
- if (! isset($id[$identifier])) {
- throw ORMException::missingIdentifierField($class->name, $identifier);
- }
- $sortedId[$identifier] = $id[$identifier];
- unset($id[$identifier]);
- }
- if ($id) {
- throw ORMException::unrecognizedIdentifierFields($class->name, array_keys($id));
- }
- $unitOfWork = $this->getUnitOfWork();
- $entity = $unitOfWork->tryGetById($sortedId, $class->rootEntityName);
- // Check identity map first
- if ($entity !== false) {
- if (! ($entity instanceof $class->name)) {
- return null;
- }
- switch (true) {
- case $lockMode === LockMode::OPTIMISTIC:
- $this->lock($entity, $lockMode, $lockVersion);
- break;
- case $lockMode === LockMode::NONE:
- case $lockMode === LockMode::PESSIMISTIC_READ:
- case $lockMode === LockMode::PESSIMISTIC_WRITE:
- $persister = $unitOfWork->getEntityPersister($class->name);
- $persister->refresh($sortedId, $entity, $lockMode);
- break;
- }
- return $entity; // Hit!
- }
- $persister = $unitOfWork->getEntityPersister($class->name);
- switch (true) {
- case $lockMode === LockMode::OPTIMISTIC:
- $entity = $persister->load($sortedId);
- $unitOfWork->lock($entity, $lockMode, $lockVersion);
- return $entity;
- case $lockMode === LockMode::PESSIMISTIC_READ:
- case $lockMode === LockMode::PESSIMISTIC_WRITE:
- return $persister->load($sortedId, null, null, [], $lockMode);
- default:
- return $persister->loadById($sortedId);
- }
- }
- /**
- * {@inheritDoc}
- */
- public function getReference($entityName, $id)
- {
- $class = $this->metadataFactory->getMetadataFor(ltrim($entityName, '\\'));
- if (! is_array($id)) {
- $id = [$class->identifier[0] => $id];
- }
- $sortedId = [];
- foreach ($class->identifier as $identifier) {
- if (! isset($id[$identifier])) {
- throw ORMException::missingIdentifierField($class->name, $identifier);
- }
- $sortedId[$identifier] = $id[$identifier];
- unset($id[$identifier]);
- }
- if ($id) {
- throw ORMException::unrecognizedIdentifierFields($class->name, array_keys($id));
- }
- $entity = $this->unitOfWork->tryGetById($sortedId, $class->rootEntityName);
- // Check identity map first, if its already in there just return it.
- if ($entity !== false) {
- return $entity instanceof $class->name ? $entity : null;
- }
- if ($class->subClasses) {
- return $this->find($entityName, $sortedId);
- }
- $entity = $this->proxyFactory->getProxy($class->name, $sortedId);
- $this->unitOfWork->registerManaged($entity, $sortedId, []);
- return $entity;
- }
- /**
- * {@inheritDoc}
- */
- public function getPartialReference($entityName, $identifier)
- {
- $class = $this->metadataFactory->getMetadataFor(ltrim($entityName, '\\'));
- $entity = $this->unitOfWork->tryGetById($identifier, $class->rootEntityName);
- // Check identity map first, if its already in there just return it.
- if ($entity !== false) {
- return $entity instanceof $class->name ? $entity : null;
- }
- if (! is_array($identifier)) {
- $identifier = [$class->identifier[0] => $identifier];
- }
- $entity = $class->newInstance();
- $class->setIdentifierValues($entity, $identifier);
- $this->unitOfWork->registerManaged($entity, $identifier, []);
- $this->unitOfWork->markReadOnly($entity);
- return $entity;
- }
- /**
- * Clears the EntityManager. All entities that are currently managed
- * by this EntityManager become detached.
- *
- * @param string|null $entityName if given, only entities of this type will get detached
- *
- * @return void
- *
- * @throws ORMInvalidArgumentException If a non-null non-string value is given.
- * @throws MappingException If a $entityName is given, but that entity is not
- * found in the mappings.
- */
- public function clear($entityName = null)
- {
- if ($entityName !== null && ! is_string($entityName)) {
- throw ORMInvalidArgumentException::invalidEntityName($entityName);
- }
- if ($entityName !== null) {
- @trigger_error(
- 'Calling ' . __METHOD__ . '() with any arguments to clear specific entities is deprecated and will not be supported in Doctrine ORM 3.0.',
- E_USER_DEPRECATED
- );
- }
- $this->unitOfWork->clear(
- $entityName === null
- ? null
- : $this->metadataFactory->getMetadataFor($entityName)->getName()
- );
- }
- /**
- * {@inheritDoc}
- */
- 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 $entity The instance to make managed and persistent.
- *
- * @return void
- *
- * @throws ORMInvalidArgumentException
- * @throws ORMException
- */
- public function persist($entity)
- {
- if (! is_object($entity)) {
- throw ORMInvalidArgumentException::invalidObject('EntityManager#persist()', $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.
- *
- * @return void
- *
- * @throws ORMInvalidArgumentException
- * @throws ORMException
- */
- public function remove($entity)
- {
- if (! is_object($entity)) {
- throw ORMInvalidArgumentException::invalidObject('EntityManager#remove()', $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.
- *
- * @return void
- *
- * @throws ORMInvalidArgumentException
- * @throws ORMException
- */
- public function refresh($entity)
- {
- if (! is_object($entity)) {
- throw ORMInvalidArgumentException::invalidObject('EntityManager#refresh()', $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.
- *
- * @deprecated 2.7 This method is being removed from the ORM and won't have any replacement
- *
- * @param object $entity The entity to detach.
- *
- * @return void
- *
- * @throws ORMInvalidArgumentException
- */
- public function detach($entity)
- {
- @trigger_error('Method ' . __METHOD__ . '() is deprecated and will be removed in Doctrine ORM 3.0.', E_USER_DEPRECATED);
- if (! is_object($entity)) {
- throw ORMInvalidArgumentException::invalidObject('EntityManager#detach()', $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.
- *
- * @deprecated 2.7 This method is being removed from the ORM and won't have any replacement
- *
- * @param object $entity The detached entity to merge into the persistence context.
- *
- * @return object The managed copy of the entity.
- *
- * @throws ORMInvalidArgumentException
- * @throws ORMException
- */
- public function merge($entity)
- {
- @trigger_error('Method ' . __METHOD__ . '() is deprecated and will be removed in Doctrine ORM 3.0.', E_USER_DEPRECATED);
- if (! is_object($entity)) {
- throw ORMInvalidArgumentException::invalidObject('EntityManager#merge()', $entity);
- }
- $this->errorIfClosed();
- return $this->unitOfWork->merge($entity);
- }
- /**
- * {@inheritDoc}
- */
- public function copy($entity, $deep = false)
- {
- @trigger_error('Method ' . __METHOD__ . '() is deprecated and will be removed in Doctrine ORM 3.0.', E_USER_DEPRECATED);
- throw new BadMethodCallException('Not implemented.');
- }
- /**
- * {@inheritDoc}
- */
- 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 ObjectRepository|EntityRepository The repository class.
- *
- * @template T
- * @psalm-param class-string<T> $entityName
- * @psalm-return EntityRepository<T>
- */
- public function getRepository($entityName)
- {
- return $this->repositoryFactory->getRepository($this, $entityName);
- }
- /**
- * Determines whether an entity instance is managed in this EntityManager.
- *
- * @param object $entity
- *
- * @return bool 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);
- }
- /**
- * {@inheritDoc}
- */
- public function getEventManager()
- {
- return $this->eventManager;
- }
- /**
- * {@inheritDoc}
- */
- public function getConfiguration()
- {
- return $this->config;
- }
- /**
- * Throws an exception if the EntityManager is closed or currently not active.
- *
- * @return void
- *
- * @throws ORMException If the EntityManager is closed.
- */
- private function errorIfClosed()
- {
- if ($this->closed) {
- throw ORMException::entityManagerClosed();
- }
- }
- /**
- * {@inheritDoc}
- */
- public function isOpen()
- {
- return ! $this->closed;
- }
- /**
- * {@inheritDoc}
- */
- public function getUnitOfWork()
- {
- return $this->unitOfWork;
- }
- /**
- * {@inheritDoc}
- */
- public function getHydrator($hydrationMode)
- {
- return $this->newHydrator($hydrationMode);
- }
- /**
- * {@inheritDoc}
- */
- public function newHydrator($hydrationMode)
- {
- switch ($hydrationMode) {
- case Query::HYDRATE_OBJECT:
- return new Internal\Hydration\ObjectHydrator($this);
- case Query::HYDRATE_ARRAY:
- return new Internal\Hydration\ArrayHydrator($this);
- case Query::HYDRATE_SCALAR:
- return new Internal\Hydration\ScalarHydrator($this);
- case Query::HYDRATE_SINGLE_SCALAR:
- return new Internal\Hydration\SingleScalarHydrator($this);
- case Query::HYDRATE_SIMPLEOBJECT:
- return new Internal\Hydration\SimpleObjectHydrator($this);
- default:
- $class = $this->config->getCustomHydrationMode($hydrationMode);
- if ($class !== null) {
- return new $class($this);
- }
- }
- throw ORMException::invalidHydrationMode($hydrationMode);
- }
- /**
- * {@inheritDoc}
- */
- public function getProxyFactory()
- {
- return $this->proxyFactory;
- }
- /**
- * {@inheritDoc}
- */
- public function initializeObject($obj)
- {
- $this->unitOfWork->initializeObject($obj);
- }
- /**
- * Factory method to create EntityManager instances.
- *
- * @param array<string, mixed>|Connection $connection 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.
- *
- * @throws InvalidArgumentException
- * @throws ORMException
- */
- public static function create($connection, Configuration $config, ?EventManager $eventManager = null)
- {
- if (! $config->getMetadataDriverImpl()) {
- throw ORMException::missingMappingDriverImpl();
- }
- $connection = static::createConnection($connection, $config, $eventManager);
- return new EntityManager($connection, $config, $connection->getEventManager());
- }
- /**
- * Factory method to create Connection instances.
- *
- * @param array<string, mixed>|Connection $connection 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 Connection
- *
- * @throws InvalidArgumentException
- * @throws ORMException
- */
- protected static function createConnection($connection, Configuration $config, ?EventManager $eventManager = null)
- {
- if (is_array($connection)) {
- return DriverManager::getConnection($connection, $config, $eventManager ?: new EventManager());
- }
- if (! $connection instanceof Connection) {
- throw new InvalidArgumentException(
- sprintf(
- 'Invalid $connection argument of type %s given%s.',
- is_object($connection) ? get_class($connection) : gettype($connection),
- is_object($connection) ? '' : ': "' . $connection . '"'
- )
- );
- }
- if ($eventManager !== null && $connection->getEventManager() !== $eventManager) {
- throw ORMException::mismatchedEventManager();
- }
- return $connection;
- }
- /**
- * {@inheritDoc}
- */
- public function getFilters()
- {
- if ($this->filterCollection === null) {
- $this->filterCollection = new FilterCollection($this);
- }
- return $this->filterCollection;
- }
- /**
- * {@inheritDoc}
- */
- public function isFiltersStateClean()
- {
- return $this->filterCollection === null || $this->filterCollection->isClean();
- }
- /**
- * {@inheritDoc}
- */
- public function hasFilters()
- {
- return $this->filterCollection !== null;
- }
- /**
- * @throws OptimisticLockException
- * @throws TransactionRequiredException
- */
- private function checkLockRequirements(int $lockMode, ClassMetadata $class): void
- {
- switch ($lockMode) {
- case LockMode::OPTIMISTIC:
- if (! $class->isVersioned) {
- throw OptimisticLockException::notVersioned($class->name);
- }
- break;
- case LockMode::PESSIMISTIC_READ:
- case LockMode::PESSIMISTIC_WRITE:
- if (! $this->getConnection()->isTransactionActive()) {
- throw TransactionRequiredException::transactionRequired();
- }
- }
- }
- }
|