vendor/doctrine/orm/lib/Doctrine/ORM/Query.php line 325

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. namespace Doctrine\ORM;
  4. use Doctrine\Common\Cache\Cache;
  5. use Doctrine\Common\Cache\Psr6\CacheAdapter;
  6. use Doctrine\Common\Cache\Psr6\DoctrineProvider;
  7. use Doctrine\Common\Collections\ArrayCollection;
  8. use Doctrine\DBAL\Cache\QueryCacheProfile;
  9. use Doctrine\DBAL\LockMode;
  10. use Doctrine\DBAL\Types\Type;
  11. use Doctrine\Deprecations\Deprecation;
  12. use Doctrine\ORM\Internal\Hydration\IterableResult;
  13. use Doctrine\ORM\Mapping\ClassMetadata;
  14. use Doctrine\ORM\Query\AST\DeleteStatement;
  15. use Doctrine\ORM\Query\AST\SelectStatement;
  16. use Doctrine\ORM\Query\AST\UpdateStatement;
  17. use Doctrine\ORM\Query\Exec\AbstractSqlExecutor;
  18. use Doctrine\ORM\Query\Parameter;
  19. use Doctrine\ORM\Query\ParameterTypeInferer;
  20. use Doctrine\ORM\Query\Parser;
  21. use Doctrine\ORM\Query\ParserResult;
  22. use Doctrine\ORM\Query\QueryException;
  23. use Doctrine\ORM\Query\ResultSetMapping;
  24. use Doctrine\ORM\Utility\HierarchyDiscriminatorResolver;
  25. use Psr\Cache\CacheItemPoolInterface;
  26. use function array_keys;
  27. use function array_values;
  28. use function assert;
  29. use function count;
  30. use function get_debug_type;
  31. use function in_array;
  32. use function is_int;
  33. use function ksort;
  34. use function md5;
  35. use function method_exists;
  36. use function reset;
  37. use function serialize;
  38. use function sha1;
  39. use function stripos;
  40. /**
  41.  * A Query object represents a DQL query.
  42.  */
  43. final class Query extends AbstractQuery
  44. {
  45.     /**
  46.      * A query object is in CLEAN state when it has NO unparsed/unprocessed DQL parts.
  47.      */
  48.     public const STATE_CLEAN 1;
  49.     /**
  50.      * A query object is in state DIRTY when it has DQL parts that have not yet been
  51.      * parsed/processed. This is automatically defined as DIRTY when addDqlQueryPart
  52.      * is called.
  53.      */
  54.     public const STATE_DIRTY 2;
  55.     /* Query HINTS */
  56.     /**
  57.      * The refresh hint turns any query into a refresh query with the result that
  58.      * any local changes in entities are overridden with the fetched values.
  59.      */
  60.     public const HINT_REFRESH 'doctrine.refresh';
  61.     public const HINT_CACHE_ENABLED 'doctrine.cache.enabled';
  62.     public const HINT_CACHE_EVICT 'doctrine.cache.evict';
  63.     /**
  64.      * Internal hint: is set to the proxy entity that is currently triggered for loading
  65.      */
  66.     public const HINT_REFRESH_ENTITY 'doctrine.refresh.entity';
  67.     /**
  68.      * The forcePartialLoad query hint forces a particular query to return
  69.      * partial objects.
  70.      *
  71.      * @todo Rename: HINT_OPTIMIZE
  72.      */
  73.     public const HINT_FORCE_PARTIAL_LOAD 'doctrine.forcePartialLoad';
  74.     /**
  75.      * The includeMetaColumns query hint causes meta columns like foreign keys and
  76.      * discriminator columns to be selected and returned as part of the query result.
  77.      *
  78.      * This hint does only apply to non-object queries.
  79.      */
  80.     public const HINT_INCLUDE_META_COLUMNS 'doctrine.includeMetaColumns';
  81.     /**
  82.      * An array of class names that implement \Doctrine\ORM\Query\TreeWalker and
  83.      * are iterated and executed after the DQL has been parsed into an AST.
  84.      */
  85.     public const HINT_CUSTOM_TREE_WALKERS 'doctrine.customTreeWalkers';
  86.     /**
  87.      * A string with a class name that implements \Doctrine\ORM\Query\TreeWalker
  88.      * and is used for generating the target SQL from any DQL AST tree.
  89.      */
  90.     public const HINT_CUSTOM_OUTPUT_WALKER 'doctrine.customOutputWalker';
  91.     /**
  92.      * Marks queries as creating only read only objects.
  93.      *
  94.      * If the object retrieved from the query is already in the identity map
  95.      * then it does not get marked as read only if it wasn't already.
  96.      */
  97.     public const HINT_READ_ONLY 'doctrine.readOnly';
  98.     public const HINT_INTERNAL_ITERATION 'doctrine.internal.iteration';
  99.     public const HINT_LOCK_MODE 'doctrine.lockMode';
  100.     /**
  101.      * The current state of this query.
  102.      *
  103.      * @var int
  104.      * @psalm-var self::STATE_*
  105.      */
  106.     private $_state self::STATE_DIRTY;
  107.     /**
  108.      * A snapshot of the parameter types the query was parsed with.
  109.      *
  110.      * @var array<string,Type>
  111.      */
  112.     private $parsedTypes = [];
  113.     /**
  114.      * Cached DQL query.
  115.      *
  116.      * @var string|null
  117.      */
  118.     private $dql null;
  119.     /**
  120.      * The parser result that holds DQL => SQL information.
  121.      *
  122.      * @var ParserResult
  123.      */
  124.     private $parserResult;
  125.     /**
  126.      * The first result to return (the "offset").
  127.      *
  128.      * @var int
  129.      */
  130.     private $firstResult 0;
  131.     /**
  132.      * The maximum number of results to return (the "limit").
  133.      *
  134.      * @var int|null
  135.      */
  136.     private $maxResults null;
  137.     /**
  138.      * The cache driver used for caching queries.
  139.      *
  140.      * @var CacheItemPoolInterface|null
  141.      */
  142.     private $queryCache;
  143.     /**
  144.      * Whether or not expire the query cache.
  145.      *
  146.      * @var bool
  147.      */
  148.     private $expireQueryCache false;
  149.     /**
  150.      * The query cache lifetime.
  151.      *
  152.      * @var int|null
  153.      */
  154.     private $queryCacheTTL;
  155.     /**
  156.      * Whether to use a query cache, if available. Defaults to TRUE.
  157.      *
  158.      * @var bool
  159.      */
  160.     private $useQueryCache true;
  161.     /**
  162.      * Gets the SQL query/queries that correspond to this DQL query.
  163.      *
  164.      * @return list<string>|string The built sql query or an array of all sql queries.
  165.      */
  166.     public function getSQL()
  167.     {
  168.         return $this->parse()->getSqlExecutor()->getSqlStatements();
  169.     }
  170.     /**
  171.      * Returns the corresponding AST for this DQL query.
  172.      *
  173.      * @return SelectStatement|UpdateStatement|DeleteStatement
  174.      */
  175.     public function getAST()
  176.     {
  177.         $parser = new Parser($this);
  178.         return $parser->getAST();
  179.     }
  180.     /**
  181.      * {@inheritdoc}
  182.      *
  183.      * @return ResultSetMapping
  184.      */
  185.     protected function getResultSetMapping()
  186.     {
  187.         // parse query or load from cache
  188.         if ($this->_resultSetMapping === null) {
  189.             $this->_resultSetMapping $this->parse()->getResultSetMapping();
  190.         }
  191.         return $this->_resultSetMapping;
  192.     }
  193.     /**
  194.      * Parses the DQL query, if necessary, and stores the parser result.
  195.      *
  196.      * Note: Populates $this->_parserResult as a side-effect.
  197.      */
  198.     private function parse(): ParserResult
  199.     {
  200.         $types = [];
  201.         foreach ($this->parameters as $parameter) {
  202.             /** @var Query\Parameter $parameter */
  203.             $types[$parameter->getName()] = $parameter->getType();
  204.         }
  205.         // Return previous parser result if the query and the filter collection are both clean
  206.         if ($this->_state === self::STATE_CLEAN && $this->parsedTypes === $types && $this->_em->isFiltersStateClean()) {
  207.             return $this->parserResult;
  208.         }
  209.         $this->_state      self::STATE_CLEAN;
  210.         $this->parsedTypes $types;
  211.         $queryCache $this->queryCache ?? $this->_em->getConfiguration()->getQueryCache();
  212.         // Check query cache.
  213.         if (! ($this->useQueryCache && $queryCache)) {
  214.             $parser = new Parser($this);
  215.             $this->parserResult $parser->parse();
  216.             return $this->parserResult;
  217.         }
  218.         $cacheItem $queryCache->getItem($this->getQueryCacheId());
  219.         if (! $this->expireQueryCache && $cacheItem->isHit()) {
  220.             $cached $cacheItem->get();
  221.             if ($cached instanceof ParserResult) {
  222.                 // Cache hit.
  223.                 $this->parserResult $cached;
  224.                 return $this->parserResult;
  225.             }
  226.         }
  227.         // Cache miss.
  228.         $parser = new Parser($this);
  229.         $this->parserResult $parser->parse();
  230.         $queryCache->save($cacheItem->set($this->parserResult)->expiresAfter($this->queryCacheTTL));
  231.         return $this->parserResult;
  232.     }
  233.     /**
  234.      * {@inheritdoc}
  235.      */
  236.     protected function _doExecute()
  237.     {
  238.         $executor $this->parse()->getSqlExecutor();
  239.         if ($this->_queryCacheProfile) {
  240.             $executor->setQueryCacheProfile($this->_queryCacheProfile);
  241.         } else {
  242.             $executor->removeQueryCacheProfile();
  243.         }
  244.         if ($this->_resultSetMapping === null) {
  245.             $this->_resultSetMapping $this->parserResult->getResultSetMapping();
  246.         }
  247.         // Prepare parameters
  248.         $paramMappings $this->parserResult->getParameterMappings();
  249.         $paramCount    count($this->parameters);
  250.         $mappingCount  count($paramMappings);
  251.         if ($paramCount $mappingCount) {
  252.             throw QueryException::tooManyParameters($mappingCount$paramCount);
  253.         }
  254.         if ($paramCount $mappingCount) {
  255.             throw QueryException::tooFewParameters($mappingCount$paramCount);
  256.         }
  257.         // evict all cache for the entity region
  258.         if ($this->hasCache && isset($this->_hints[self::HINT_CACHE_EVICT]) && $this->_hints[self::HINT_CACHE_EVICT]) {
  259.             $this->evictEntityCacheRegion();
  260.         }
  261.         [$sqlParams$types] = $this->processParameterMappings($paramMappings);
  262.         $this->evictResultSetCache(
  263.             $executor,
  264.             $sqlParams,
  265.             $types,
  266.             $this->_em->getConnection()->getParams()
  267.         );
  268.         return $executor->execute($this->_em->getConnection(), $sqlParams$types);
  269.     }
  270.     /**
  271.      * @param array<string,mixed> $sqlParams
  272.      * @param array<string,Type>  $types
  273.      * @param array<string,mixed> $connectionParams
  274.      */
  275.     private function evictResultSetCache(
  276.         AbstractSqlExecutor $executor,
  277.         array $sqlParams,
  278.         array $types,
  279.         array $connectionParams
  280.     ): void {
  281.         if ($this->_queryCacheProfile === null || ! $this->getExpireResultCache()) {
  282.             return;
  283.         }
  284.         $cache method_exists(QueryCacheProfile::class, 'getResultCache')
  285.             ? $this->_queryCacheProfile->getResultCache()
  286.             : $this->_queryCacheProfile->getResultCacheDriver();
  287.         assert($cache !== null);
  288.         $statements = (array) $executor->getSqlStatements(); // Type casted since it can either be a string or an array
  289.         foreach ($statements as $statement) {
  290.             $cacheKeys $this->_queryCacheProfile->generateCacheKeys($statement$sqlParams$types$connectionParams);
  291.             $cache instanceof CacheItemPoolInterface
  292.                 $cache->deleteItem(reset($cacheKeys))
  293.                 : $cache->delete(reset($cacheKeys));
  294.         }
  295.     }
  296.     /**
  297.      * Evict entity cache region
  298.      */
  299.     private function evictEntityCacheRegion(): void
  300.     {
  301.         $AST $this->getAST();
  302.         if ($AST instanceof SelectStatement) {
  303.             throw new QueryException('The hint "HINT_CACHE_EVICT" is not valid for select statements.');
  304.         }
  305.         $className $AST instanceof DeleteStatement
  306.             $AST->deleteClause->abstractSchemaName
  307.             $AST->updateClause->abstractSchemaName;
  308.         $this->_em->getCache()->evictEntityRegion($className);
  309.     }
  310.     /**
  311.      * Processes query parameter mappings.
  312.      *
  313.      * @param array<list<int>> $paramMappings
  314.      *
  315.      * @return mixed[][]
  316.      * @psalm-return array{0: list<mixed>, 1: array}
  317.      *
  318.      * @throws Query\QueryException
  319.      */
  320.     private function processParameterMappings(array $paramMappings): array
  321.     {
  322.         $sqlParams = [];
  323.         $types     = [];
  324.         foreach ($this->parameters as $parameter) {
  325.             $key $parameter->getName();
  326.             if (! isset($paramMappings[$key])) {
  327.                 throw QueryException::unknownParameter($key);
  328.             }
  329.             [$value$type] = $this->resolveParameterValue($parameter);
  330.             foreach ($paramMappings[$key] as $position) {
  331.                 $types[$position] = $type;
  332.             }
  333.             $sqlPositions $paramMappings[$key];
  334.             // optimized multi value sql positions away for now,
  335.             // they are not allowed in DQL anyways.
  336.             $value      = [$value];
  337.             $countValue count($value);
  338.             for ($i 0$l count($sqlPositions); $i $l$i++) {
  339.                 $sqlParams[$sqlPositions[$i]] = $value[$i $countValue];
  340.             }
  341.         }
  342.         if (count($sqlParams) !== count($types)) {
  343.             throw QueryException::parameterTypeMismatch();
  344.         }
  345.         if ($sqlParams) {
  346.             ksort($sqlParams);
  347.             $sqlParams array_values($sqlParams);
  348.             ksort($types);
  349.             $types array_values($types);
  350.         }
  351.         return [$sqlParams$types];
  352.     }
  353.     /**
  354.      * @return mixed[] tuple of (value, type)
  355.      * @psalm-return array{0: mixed, 1: mixed}
  356.      */
  357.     private function resolveParameterValue(Parameter $parameter): array
  358.     {
  359.         if ($parameter->typeWasSpecified()) {
  360.             return [$parameter->getValue(), $parameter->getType()];
  361.         }
  362.         $key           $parameter->getName();
  363.         $originalValue $parameter->getValue();
  364.         $value         $originalValue;
  365.         $rsm           $this->getResultSetMapping();
  366.         if ($value instanceof ClassMetadata && isset($rsm->metadataParameterMapping[$key])) {
  367.             $value $value->getMetadataValue($rsm->metadataParameterMapping[$key]);
  368.         }
  369.         if ($value instanceof ClassMetadata && isset($rsm->discriminatorParameters[$key])) {
  370.             $value array_keys(HierarchyDiscriminatorResolver::resolveDiscriminatorsForClass($value$this->_em));
  371.         }
  372.         $processedValue $this->processParameterValue($value);
  373.         return [
  374.             $processedValue,
  375.             $originalValue === $processedValue
  376.                 $parameter->getType()
  377.                 : ParameterTypeInferer::inferType($processedValue),
  378.         ];
  379.     }
  380.     /**
  381.      * Defines a cache driver to be used for caching queries.
  382.      *
  383.      * @deprecated Call {@see setQueryCache()} instead.
  384.      *
  385.      * @param Cache|null $queryCache Cache driver.
  386.      *
  387.      * @return $this
  388.      */
  389.     public function setQueryCacheDriver($queryCache): self
  390.     {
  391.         Deprecation::trigger(
  392.             'doctrine/orm',
  393.             'https://github.com/doctrine/orm/pull/9004',
  394.             '%s is deprecated and will be removed in Doctrine 3.0. Use setQueryCache() instead.',
  395.             __METHOD__
  396.         );
  397.         $this->queryCache $queryCache CacheAdapter::wrap($queryCache) : null;
  398.         return $this;
  399.     }
  400.     /**
  401.      * Defines a cache driver to be used for caching queries.
  402.      *
  403.      * @return $this
  404.      */
  405.     public function setQueryCache(?CacheItemPoolInterface $queryCache): self
  406.     {
  407.         $this->queryCache $queryCache;
  408.         return $this;
  409.     }
  410.     /**
  411.      * Defines whether the query should make use of a query cache, if available.
  412.      *
  413.      * @param bool $bool
  414.      *
  415.      * @return $this
  416.      */
  417.     public function useQueryCache($bool): self
  418.     {
  419.         $this->useQueryCache $bool;
  420.         return $this;
  421.     }
  422.     /**
  423.      * Returns the cache driver used for query caching.
  424.      *
  425.      * @deprecated
  426.      *
  427.      * @return Cache|null The cache driver used for query caching or NULL, if
  428.      * this Query does not use query caching.
  429.      */
  430.     public function getQueryCacheDriver(): ?Cache
  431.     {
  432.         Deprecation::trigger(
  433.             'doctrine/orm',
  434.             'https://github.com/doctrine/orm/pull/9004',
  435.             '%s is deprecated and will be removed in Doctrine 3.0 without replacement.',
  436.             __METHOD__
  437.         );
  438.         $queryCache $this->queryCache ?? $this->_em->getConfiguration()->getQueryCache();
  439.         return $queryCache DoctrineProvider::wrap($queryCache) : null;
  440.     }
  441.     /**
  442.      * Defines how long the query cache will be active before expire.
  443.      *
  444.      * @param int|null $timeToLive How long the cache entry is valid.
  445.      *
  446.      * @return $this
  447.      */
  448.     public function setQueryCacheLifetime($timeToLive): self
  449.     {
  450.         if ($timeToLive !== null) {
  451.             $timeToLive = (int) $timeToLive;
  452.         }
  453.         $this->queryCacheTTL $timeToLive;
  454.         return $this;
  455.     }
  456.     /**
  457.      * Retrieves the lifetime of resultset cache.
  458.      */
  459.     public function getQueryCacheLifetime(): ?int
  460.     {
  461.         return $this->queryCacheTTL;
  462.     }
  463.     /**
  464.      * Defines if the query cache is active or not.
  465.      *
  466.      * @param bool $expire Whether or not to force query cache expiration.
  467.      *
  468.      * @return $this
  469.      */
  470.     public function expireQueryCache($expire true): self
  471.     {
  472.         $this->expireQueryCache $expire;
  473.         return $this;
  474.     }
  475.     /**
  476.      * Retrieves if the query cache is active or not.
  477.      */
  478.     public function getExpireQueryCache(): bool
  479.     {
  480.         return $this->expireQueryCache;
  481.     }
  482.     public function free(): void
  483.     {
  484.         parent::free();
  485.         $this->dql    null;
  486.         $this->_state self::STATE_CLEAN;
  487.     }
  488.     /**
  489.      * Sets a DQL query string.
  490.      *
  491.      * @param string|null $dqlQuery DQL Query.
  492.      */
  493.     public function setDQL($dqlQuery): self
  494.     {
  495.         if ($dqlQuery === null) {
  496.             Deprecation::trigger(
  497.                 'doctrine/orm',
  498.                 'https://github.com/doctrine/orm/pull/9784',
  499.                 'Calling %s with null is deprecated and will result in a TypeError in Doctrine 3.0',
  500.                 __METHOD__
  501.             );
  502.             return $this;
  503.         }
  504.         $this->dql    $dqlQuery;
  505.         $this->_state self::STATE_DIRTY;
  506.         return $this;
  507.     }
  508.     /**
  509.      * Returns the DQL query that is represented by this query object.
  510.      */
  511.     public function getDQL(): ?string
  512.     {
  513.         return $this->dql;
  514.     }
  515.     /**
  516.      * Returns the state of this query object
  517.      * By default the type is Doctrine_ORM_Query_Abstract::STATE_CLEAN but if it appears any unprocessed DQL
  518.      * part, it is switched to Doctrine_ORM_Query_Abstract::STATE_DIRTY.
  519.      *
  520.      * @see AbstractQuery::STATE_CLEAN
  521.      * @see AbstractQuery::STATE_DIRTY
  522.      *
  523.      * @return int The query state.
  524.      * @psalm-return self::STATE_* The query state.
  525.      */
  526.     public function getState(): int
  527.     {
  528.         return $this->_state;
  529.     }
  530.     /**
  531.      * Method to check if an arbitrary piece of DQL exists
  532.      *
  533.      * @param string $dql Arbitrary piece of DQL to check for.
  534.      */
  535.     public function contains($dql): bool
  536.     {
  537.         return stripos($this->getDQL(), $dql) !== false;
  538.     }
  539.     /**
  540.      * Sets the position of the first result to retrieve (the "offset").
  541.      *
  542.      * @param int|null $firstResult The first result to return.
  543.      *
  544.      * @return $this
  545.      */
  546.     public function setFirstResult($firstResult): self
  547.     {
  548.         if (! is_int($firstResult)) {
  549.             Deprecation::trigger(
  550.                 'doctrine/orm',
  551.                 'https://github.com/doctrine/orm/pull/9809',
  552.                 'Calling %s with %s is deprecated and will result in a TypeError in Doctrine 3.0. Pass an integer.',
  553.                 __METHOD__,
  554.                 get_debug_type($firstResult)
  555.             );
  556.             $firstResult = (int) $firstResult;
  557.         }
  558.         $this->firstResult $firstResult;
  559.         $this->_state      self::STATE_DIRTY;
  560.         return $this;
  561.     }
  562.     /**
  563.      * Gets the position of the first result the query object was set to retrieve (the "offset").
  564.      * Returns NULL if {@link setFirstResult} was not applied to this query.
  565.      *
  566.      * @return int|null The position of the first result.
  567.      */
  568.     public function getFirstResult(): ?int
  569.     {
  570.         return $this->firstResult;
  571.     }
  572.     /**
  573.      * Sets the maximum number of results to retrieve (the "limit").
  574.      *
  575.      * @param int|null $maxResults
  576.      *
  577.      * @return $this
  578.      */
  579.     public function setMaxResults($maxResults): self
  580.     {
  581.         if ($maxResults !== null) {
  582.             $maxResults = (int) $maxResults;
  583.         }
  584.         $this->maxResults $maxResults;
  585.         $this->_state     self::STATE_DIRTY;
  586.         return $this;
  587.     }
  588.     /**
  589.      * Gets the maximum number of results the query object was set to retrieve (the "limit").
  590.      * Returns NULL if {@link setMaxResults} was not applied to this query.
  591.      *
  592.      * @return int|null Maximum number of results.
  593.      */
  594.     public function getMaxResults(): ?int
  595.     {
  596.         return $this->maxResults;
  597.     }
  598.     /**
  599.      * Executes the query and returns an IterableResult that can be used to incrementally
  600.      * iterated over the result.
  601.      *
  602.      * @deprecated
  603.      *
  604.      * @param ArrayCollection|mixed[]|null $parameters    The query parameters.
  605.      * @param string|int                   $hydrationMode The hydration mode to use.
  606.      * @psalm-param ArrayCollection<int, Parameter>|array<string, mixed>|null $parameters
  607.      * @psalm-param string|AbstractQuery::HYDRATE_*|null                      $hydrationMode
  608.      */
  609.     public function iterate($parameters null$hydrationMode self::HYDRATE_OBJECT): IterableResult
  610.     {
  611.         $this->setHint(self::HINT_INTERNAL_ITERATIONtrue);
  612.         return parent::iterate($parameters$hydrationMode);
  613.     }
  614.     /** {@inheritDoc} */
  615.     public function toIterable(iterable $parameters = [], $hydrationMode self::HYDRATE_OBJECT): iterable
  616.     {
  617.         $this->setHint(self::HINT_INTERNAL_ITERATIONtrue);
  618.         return parent::toIterable($parameters$hydrationMode);
  619.     }
  620.     /**
  621.      * {@inheritdoc}
  622.      */
  623.     public function setHint($name$value): self
  624.     {
  625.         $this->_state self::STATE_DIRTY;
  626.         return parent::setHint($name$value);
  627.     }
  628.     /**
  629.      * {@inheritdoc}
  630.      */
  631.     public function setHydrationMode($hydrationMode): self
  632.     {
  633.         $this->_state self::STATE_DIRTY;
  634.         return parent::setHydrationMode($hydrationMode);
  635.     }
  636.     /**
  637.      * Set the lock mode for this Query.
  638.      *
  639.      * @see \Doctrine\DBAL\LockMode
  640.      *
  641.      * @param int $lockMode
  642.      * @psalm-param LockMode::* $lockMode
  643.      *
  644.      * @return $this
  645.      *
  646.      * @throws TransactionRequiredException
  647.      */
  648.     public function setLockMode($lockMode): self
  649.     {
  650.         if (in_array($lockMode, [LockMode::NONELockMode::PESSIMISTIC_READLockMode::PESSIMISTIC_WRITE], true)) {
  651.             if (! $this->_em->getConnection()->isTransactionActive()) {
  652.                 throw TransactionRequiredException::transactionRequired();
  653.             }
  654.         }
  655.         $this->setHint(self::HINT_LOCK_MODE$lockMode);
  656.         return $this;
  657.     }
  658.     /**
  659.      * Get the current lock mode for this query.
  660.      *
  661.      * @return int|null The current lock mode of this query or NULL if no specific lock mode is set.
  662.      */
  663.     public function getLockMode(): ?int
  664.     {
  665.         $lockMode $this->getHint(self::HINT_LOCK_MODE);
  666.         if ($lockMode === false) {
  667.             return null;
  668.         }
  669.         return $lockMode;
  670.     }
  671.     /**
  672.      * Generate a cache id for the query cache - reusing the Result-Cache-Id generator.
  673.      */
  674.     protected function getQueryCacheId(): string
  675.     {
  676.         ksort($this->_hints);
  677.         return md5(
  678.             $this->getDQL() . serialize($this->_hints) .
  679.             '&platform=' get_debug_type($this->getEntityManager()->getConnection()->getDatabasePlatform()) .
  680.             ($this->_em->hasFilters() ? $this->_em->getFilters()->getHash() : '') .
  681.             '&firstResult=' $this->firstResult '&maxResult=' $this->maxResults .
  682.             '&hydrationMode=' $this->_hydrationMode '&types=' serialize($this->parsedTypes) . 'DOCTRINE_QUERY_CACHE_SALT'
  683.         );
  684.     }
  685.     protected function getHash(): string
  686.     {
  687.         return sha1(parent::getHash() . '-' $this->firstResult '-' $this->maxResults);
  688.     }
  689.     /**
  690.      * Cleanup Query resource when clone is called.
  691.      */
  692.     public function __clone()
  693.     {
  694.         parent::__clone();
  695.         $this->_state self::STATE_DIRTY;
  696.     }
  697. }