Symfony2 - 自定义 DQL 函数已注册但不起作用
Symfony2 - Custom DQL Function Registered But Does Not work
我拼命尝试在 mysql Symfony2 中包含 GREATEST 函数,但是,我仍然收到错误。
在DQL中添加了函数:
<?php
namespace DoctrineExtensions\Query\Mysql;
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Lexer;
use Doctrine\ORM\Query\Parser;
use Doctrine\ORM\Query\SqlWalker;
class Greatest extends FunctionNode
{
private $field = null;
private $values = [];
/**
* @param Parser $parser
*/
public function parse(Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->field = $parser->ArithmeticExpression();
$lexer = $parser->getLexer();
while (count($this->values) < 1 ||
$lexer->lookahead['type'] != Lexer::T_CLOSE_PARENTHESIS) {
$parser->match(Lexer::T_COMMA);
$this->values[] = $parser->ArithmeticExpression();
}
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
/**
* @param SqlWalker $sqlWalker
* @return string
*/
public function getSql(SqlWalker $sqlWalker)
{
$query = 'GREATEST(';
$query .= $this->field->dispatch($sqlWalker);
$query .= ', ';
for ($i = 0; $i < count($this->values); $i++) {
if ($i > 0) {
$query .= ', ';
}
$query .= $this->values[$i]->dispatch($sqlWalker);
}
$query .= ')';
return $query;
}
}
在Config.yml
orm:
auto_generate_proxy_classes: false
auto_mapping: true
dql:
datetime_functions:
Greatest: DoctrineExtensions\Query\Mysql\Greatest
问题:在我的存储库中执行以下代码块时,出现以下错误:
$admitObj = $em->createQueryBuilder();
$admitObj
->select('A')
->from("EntityBundle:Admit", "A")
->orderBy("GREATEST( COALESCE(A.date1, 0), COALESCE(A.date2, 0))", "DESC");
$admit = $admitObj->setMaxResults(1)->getQuery()->getResult();
错误:[语法错误] 第 0 行,第 111 列:错误:字符串预期结束,得到 '('
我错过了什么?为什么 DQL/Symfony/PDO/... 无法识别函数?非常感谢任何帮助!
经过长时间的研究,我发现这是我的 parser.php 文件中的问题。我解决了这个问题以替换我的解析器文件中的以下代码。是按顺序使用最大函数的问题。
/**
* OrderByClause ::= "ORDER" "BY" OrderByItem {"," OrderByItem}*
*
* @return \Doctrine\ORM\Query\AST\OrderByClause
*/
public function OrderByClause()
{
$this->match(Lexer::T_ORDER);
$this->match(Lexer::T_BY);
$orderByItems = array();
$orderByItems[] = $this->OrderByItem();
while ($this->lexer->isNextToken(Lexer::T_COMMA)) {
$this->match(Lexer::T_COMMA);
$orderByItems[] = $this->OrderByItem();
}
return new AST\OrderByClause($orderByItems);
}
我拼命尝试在 mysql Symfony2 中包含 GREATEST 函数,但是,我仍然收到错误。
在DQL中添加了函数:
<?php
namespace DoctrineExtensions\Query\Mysql;
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Lexer;
use Doctrine\ORM\Query\Parser;
use Doctrine\ORM\Query\SqlWalker;
class Greatest extends FunctionNode
{
private $field = null;
private $values = [];
/**
* @param Parser $parser
*/
public function parse(Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->field = $parser->ArithmeticExpression();
$lexer = $parser->getLexer();
while (count($this->values) < 1 ||
$lexer->lookahead['type'] != Lexer::T_CLOSE_PARENTHESIS) {
$parser->match(Lexer::T_COMMA);
$this->values[] = $parser->ArithmeticExpression();
}
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
/**
* @param SqlWalker $sqlWalker
* @return string
*/
public function getSql(SqlWalker $sqlWalker)
{
$query = 'GREATEST(';
$query .= $this->field->dispatch($sqlWalker);
$query .= ', ';
for ($i = 0; $i < count($this->values); $i++) {
if ($i > 0) {
$query .= ', ';
}
$query .= $this->values[$i]->dispatch($sqlWalker);
}
$query .= ')';
return $query;
}
}
在Config.yml
orm:
auto_generate_proxy_classes: false
auto_mapping: true
dql:
datetime_functions:
Greatest: DoctrineExtensions\Query\Mysql\Greatest
问题:在我的存储库中执行以下代码块时,出现以下错误:
$admitObj = $em->createQueryBuilder();
$admitObj
->select('A')
->from("EntityBundle:Admit", "A")
->orderBy("GREATEST( COALESCE(A.date1, 0), COALESCE(A.date2, 0))", "DESC");
$admit = $admitObj->setMaxResults(1)->getQuery()->getResult();
错误:[语法错误] 第 0 行,第 111 列:错误:字符串预期结束,得到 '('
我错过了什么?为什么 DQL/Symfony/PDO/... 无法识别函数?非常感谢任何帮助!
经过长时间的研究,我发现这是我的 parser.php 文件中的问题。我解决了这个问题以替换我的解析器文件中的以下代码。是按顺序使用最大函数的问题。
/**
* OrderByClause ::= "ORDER" "BY" OrderByItem {"," OrderByItem}*
*
* @return \Doctrine\ORM\Query\AST\OrderByClause
*/
public function OrderByClause()
{
$this->match(Lexer::T_ORDER);
$this->match(Lexer::T_BY);
$orderByItems = array();
$orderByItems[] = $this->OrderByItem();
while ($this->lexer->isNextToken(Lexer::T_COMMA)) {
$this->match(Lexer::T_COMMA);
$orderByItems[] = $this->OrderByItem();
}
return new AST\OrderByClause($orderByItems);
}