Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow DQL functions to specify return type #7941

Merged
merged 1 commit into from
Dec 12, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion lib/Doctrine/ORM/Query/AST/Functions/AbsFunction.php
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
/**
* "ABS" "(" SimpleArithmeticExpression ")"
*
*
*
* @link www.doctrine-project.org
* @since 2.0
* @author Guilherme Blanco <guilhermeblanco@hotmail.com>
Expand Down
9 changes: 8 additions & 1 deletion lib/Doctrine/ORM/Query/AST/Functions/CountFunction.php
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@

namespace Doctrine\ORM\Query\AST\Functions;

use Doctrine\DBAL\Types\Type;
use Doctrine\ORM\Query\AST\TypedExpression;
use Doctrine\ORM\Query\Parser;
use Doctrine\ORM\Query\SqlWalker;
use Doctrine\ORM\Query\AST\AggregateExpression;
Expand All @@ -29,7 +31,7 @@
* @since 2.6
* @author Mathew Davies <thepixeldeveloper@icloud.com>
*/
final class CountFunction extends FunctionNode
final class CountFunction extends FunctionNode implements TypedExpression
{
/**
* @var AggregateExpression
Expand All @@ -51,4 +53,9 @@ public function parse(Parser $parser): void
{
$this->aggregateExpression = $parser->AggregateExpression();
}

public function getReturnType() : Type
{
return Type::getType(Type::INTEGER);
}
}
2 changes: 1 addition & 1 deletion lib/Doctrine/ORM/Query/AST/Functions/FunctionNode.php
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
/**
* Abstract Function Node.
*
*
*
* @link www.doctrine-project.org
* @since 2.0
* @author Guilherme Blanco <guilhermeblanco@hotmail.com>
Expand Down
9 changes: 8 additions & 1 deletion lib/Doctrine/ORM/Query/AST/Functions/LengthFunction.php
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@

namespace Doctrine\ORM\Query\AST\Functions;

use Doctrine\DBAL\Types\Type;
use Doctrine\ORM\Query\AST\TypedExpression;
use Doctrine\ORM\Query\Lexer;

/**
Expand All @@ -32,7 +34,7 @@
* @author Roman Borschel <roman@code-factory.org>
* @author Benjamin Eberlei <kontakt@beberlei.de>
*/
class LengthFunction extends FunctionNode
class LengthFunction extends FunctionNode implements TypedExpression
{
public $stringPrimary;

Expand Down Expand Up @@ -60,4 +62,9 @@ public function parse(\Doctrine\ORM\Query\Parser $parser)

$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}

public function getReturnType() : Type
{
return Type::getType(Type::INTEGER);
}
}
2 changes: 1 addition & 1 deletion lib/Doctrine/ORM/Query/AST/Functions/SqrtFunction.php
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
/**
* "SQRT" "(" SimpleArithmeticExpression ")"
*
*
*
* @link www.doctrine-project.org
* @since 2.0
* @author Guilherme Blanco <guilhermeblanco@hotmail.com>
Expand Down
15 changes: 15 additions & 0 deletions lib/Doctrine/ORM/Query/AST/TypedExpression.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
<?php

declare(strict_types=1);

namespace Doctrine\ORM\Query\AST;

use Doctrine\DBAL\Types\Type;

/**
* Provides an API for resolving the type of a Node
*/
interface TypedExpression
{
public function getReturnType() : Type;
}
14 changes: 12 additions & 2 deletions lib/Doctrine/ORM/Query/SqlWalker.php
Original file line number Diff line number Diff line change
Expand Up @@ -1339,10 +1339,20 @@ public function walkSelectExpression($selectExpression)

$this->scalarResultAliasMap[$resultAlias] = $columnAlias;

if ( ! $hidden) {
// We cannot resolve field type here; assume 'string'.
if ($hidden) {
break;
}

if (! $expr instanceof Query\AST\TypedExpression) {
// Conceptually we could resolve field type here by traverse through AST to retrieve field type,
// but this is not a feasible solution; assume 'string'.
$this->rsm->addScalarResult($columnAlias, $resultAlias, 'string');

break;
}

$this->rsm->addScalarResult($columnAlias, $resultAlias, $expr->getReturnType()->getName());

break;

case ($expr instanceof AST\Subselect):
Expand Down
101 changes: 101 additions & 0 deletions tests/Doctrine/Tests/ORM/Functional/Ticket/GH7941Test.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
<?php

declare(strict_types=1);

namespace Doctrine\Tests\ORM\Functional\Ticket;

use DateTimeImmutable;
use Doctrine\Tests\OrmFunctionalTestCase;
use function ltrim;
use function strlen;

/** @group GH7941 */
final class GH7941Test extends OrmFunctionalTestCase
{
private const PRODUCTS = [
['name' => 'Test 1', 'price' => '100', 'square_root' => '/^10(\.0+)?$/'],
['name' => 'Test 2', 'price' => '100', 'square_root' => '/^10(\.0+)?$/'],
['name' => 'Test 3', 'price' => '100', 'square_root' => '/^10(\.0+)?$/'],
['name' => 'Test 4', 'price' => '25', 'square_root' => '/^5(\.0+)?$/'],
['name' => 'Test 5', 'price' => '25', 'square_root' => '/^5(\.0+)?$/'],
['name' => 'Test 6', 'price' => '-25', 'square_root' => '/^5(\.0+)?$/'],
];

protected function setUp() : void
{
parent::setUp();

$this->setUpEntitySchema([GH7941Product::class]);

foreach (self::PRODUCTS as $product) {
$this->_em->persist(new GH7941Product($product['name'], $product['price']));
}

$this->_em->flush();
$this->_em->clear();
}

/** @test */
public function typesShouldBeConvertedForDQLFunctions() : void
{
$query = $this->_em->createQuery(
'SELECT
COUNT(product.id) as count,
SUM(product.price) as sales,
AVG(product.price) as average
FROM ' . GH7941Product::class . ' product'
);

$result = $query->getSingleResult();

self::assertSame(6, $result['count']);
self::assertSame('325', $result['sales']);
self::assertRegExp('/^54\.16+7$/', $result['average']);
Comment on lines +41 to +53
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@Grafikart this is what I meant with a more functional approach 😄

The main advantage is that we only rely on the public API here, which makes it easier to detect BC-breaks.

By using self::assertSame() we check that both value and type match expected value and self::assertRegExp() only accepts strings.


$query = $this->_em->createQuery(
'SELECT
ABS(product.price) as absolute,
SQRT(ABS(product.price)) as square_root,
LENGTH(product.name) as length
FROM ' . GH7941Product::class . ' product'
);

foreach ($query->getResult() as $i => $item) {
$product = self::PRODUCTS[$i];

self::assertSame(ltrim($product['price'], '-'), $item['absolute']);
self::assertSame(strlen($product['name']), $item['length']);
self::assertRegExp($product['square_root'], $item['square_root']);
}
}
}

/**
* @Entity
* @Table()
*/
class GH7941Product
{
/**
* @Id
* @GeneratedValue
* @Column(type="integer")
*/
public $id;

/** @Column(type="string") */
public $name;

/** @Column(type="decimal") */
public $price;

/** @Column(type="datetime_immutable") */
public $createdAt;

public function __construct(string $name, string $price)
{
$this->name = $name;
$this->price = $price;
$this->createdAt = new DateTimeImmutable();
}
}