From 517cb7e2a2aad66a1172c89fcfb4c9c6170b4ed5 Mon Sep 17 00:00:00 2001 From: Benjamin Morel Date: Tue, 17 Mar 2015 17:53:34 +0100 Subject: [PATCH] Honor convertToDatabaseValueSQL() in DQL query parameters --- lib/Doctrine/ORM/Query.php | 25 ++++-- lib/Doctrine/ORM/Query/SqlWalker.php | 6 ++ .../ORM/Functional/Ticket/DDC2224Test.php | 88 +++++++++++++++++++ 3 files changed, 114 insertions(+), 5 deletions(-) create mode 100644 tests/Doctrine/Tests/ORM/Functional/Ticket/DDC2224Test.php diff --git a/lib/Doctrine/ORM/Query.php b/lib/Doctrine/ORM/Query.php index 6839b480d..e341c1a2b 100644 --- a/lib/Doctrine/ORM/Query.php +++ b/lib/Doctrine/ORM/Query.php @@ -130,6 +130,13 @@ final class Query extends AbstractQuery */ private $_state = self::STATE_CLEAN; + /** + * A snapshot of the parameter types the query was parsed with. + * + * @var array + */ + private $_parsedTypes = array(); + /** * Cached DQL query. * @@ -234,12 +241,20 @@ final class Query extends AbstractQuery */ private function _parse() { + $types = array(); + + foreach ($this->parameters as $parameter) { + /** @var Query\Parameter $parameter */ + $types[$parameter->getName()] = $parameter->getType(); + } + // Return previous parser result if the query and the filter collection are both clean - if ($this->_state === self::STATE_CLEAN && $this->_em->isFiltersStateClean()) { + if ($this->_state === self::STATE_CLEAN && $this->_parsedTypes === $types && $this->_em->isFiltersStateClean()) { return $this->_parserResult; } $this->_state = self::STATE_CLEAN; + $this->_parsedTypes = $types; // Check query cache. if ( ! ($this->_useQueryCache && ($queryCache = $this->getQueryCacheDriver()))) { @@ -250,7 +265,7 @@ final class Query extends AbstractQuery return $this->_parserResult; } - $hash = $this->_getQueryCacheId(); + $hash = $this->_getQueryCacheId($types); $cached = $this->_expireQueryCache ? false : $queryCache->fetch($hash); if ($cached instanceof ParserResult) { @@ -678,11 +693,11 @@ final class Query extends AbstractQuery /** * Generate a cache id for the query cache - reusing the Result-Cache-Id generator. * - * The query cache + * @param array $types The parameter types, indexed by parameter key. * * @return string */ - protected function _getQueryCacheId() + protected function _getQueryCacheId(array $types) { ksort($this->_hints); @@ -696,7 +711,7 @@ final class Query extends AbstractQuery '&platform=' . $platform . ($this->_em->hasFilters() ? $this->_em->getFilters()->getHash() : '') . '&firstResult=' . $this->_firstResult . '&maxResult=' . $this->_maxResults . - '&hydrationMode='.$this->_hydrationMode.'DOCTRINE_QUERY_CACHE_SALT' + '&hydrationMode=' . $this->_hydrationMode . '&types=' . serialize($types) . 'DOCTRINE_QUERY_CACHE_SALT' ); } diff --git a/lib/Doctrine/ORM/Query/SqlWalker.php b/lib/Doctrine/ORM/Query/SqlWalker.php index 486d4af62..3c4fc8ed1 100644 --- a/lib/Doctrine/ORM/Query/SqlWalker.php +++ b/lib/Doctrine/ORM/Query/SqlWalker.php @@ -2203,6 +2203,12 @@ class SqlWalker implements TreeWalker { $this->parserResult->addParameterMapping($inputParam->name, $this->sqlParamIndex++); + $parameter = $this->query->getParameter($inputParam->name); + + if ($parameter && Type::hasType($type = $parameter->getType())) { + return Type::getType($type)->convertToDatabaseValueSQL('?', $this->platform); + } + return '?'; } diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC2224Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC2224Test.php new file mode 100644 index 000000000..781d644ff --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC2224Test.php @@ -0,0 +1,88 @@ +_em->createQuery($dql); + $query->setQueryCacheDriver(new ArrayCache()); + + $query->setParameter('field', 'test', 'DDC2224Type'); + $this->assertStringEndsWith('.field = FUNCTION(?)', $query->getSQL()); + + return $query; + } + + /** + * @depends testIssue + */ + public function testCacheMissWhenTypeChanges(Query $query) + { + $query->setParameter('field', 'test', 'string'); + $this->assertStringEndsWith('.field = ?', $query->getSQL()); + } +} + +class DDC2224Type extends Type +{ + /** + * {@inheritdoc} + */ + public function getSQLDeclaration(array $fieldDeclaration, AbstractPlatform $platform) + { + return $platform->getVarcharTypeDeclarationSQL($fieldDeclaration); + } + + public function getName() + { + return 'DDC2224Type'; + } + + /** + * {@inheritdoc} + */ + public function canRequireSQLConversion() + { + return true; + } + + /** + * {@inheritdoc} + */ + public function convertToDatabaseValueSQL($sqlExpr, AbstractPlatform $platform) + { + return sprintf('FUNCTION(%s)', $sqlExpr); + } +} + +/** + * @Entity + */ +class DDC2224Entity +{ + /** + * @Id @GeneratedValue @Column(type="integer") + */ + public $id; + + /** + * @Column(type="DDC2224Type") + */ + public $field; +}