2012-08-19 21:58:04 +04:00
|
|
|
<?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\Query;
|
|
|
|
|
|
|
|
use Doctrine\Common\Collections\ArrayCollection;
|
|
|
|
|
|
|
|
use Doctrine\Common\Collections\Expr\ExpressionVisitor;
|
|
|
|
use Doctrine\Common\Collections\Expr\Comparison;
|
|
|
|
use Doctrine\Common\Collections\Expr\CompositeExpression;
|
|
|
|
use Doctrine\Common\Collections\Expr\Value;
|
|
|
|
|
|
|
|
use Doctrine\ORM\Query\Parameter;
|
|
|
|
|
|
|
|
/**
|
2012-12-13 16:36:14 +00:00
|
|
|
* Converts Collection expressions to Query expressions.
|
2012-08-19 21:58:04 +04:00
|
|
|
*
|
|
|
|
* @author Kirill chEbba Chebunin <iam@chebba.org>
|
2012-08-29 16:00:02 +04:00
|
|
|
* @since 2.4
|
2012-08-19 21:58:04 +04:00
|
|
|
*/
|
|
|
|
class QueryExpressionVisitor extends ExpressionVisitor
|
|
|
|
{
|
2012-12-13 16:36:14 +00:00
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
2012-08-19 21:58:04 +04:00
|
|
|
private static $operatorMap = array(
|
|
|
|
Comparison::GT => Expr\Comparison::GT,
|
|
|
|
Comparison::GTE => Expr\Comparison::GTE,
|
|
|
|
Comparison::LT => Expr\Comparison::LT,
|
|
|
|
Comparison::LTE => Expr\Comparison::LTE
|
|
|
|
);
|
|
|
|
|
2013-11-01 13:43:03 +01:00
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private $rootAlias;
|
|
|
|
|
2012-12-13 16:36:14 +00:00
|
|
|
/**
|
|
|
|
* @var Expr
|
|
|
|
*/
|
2012-08-19 21:58:04 +04:00
|
|
|
private $expr;
|
2012-12-13 16:36:14 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
2012-08-20 19:54:11 +04:00
|
|
|
private $parameters = array();
|
2012-08-19 21:58:04 +04:00
|
|
|
|
|
|
|
/**
|
2013-11-01 13:43:03 +01:00
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
* @param string $rootAlias
|
2012-08-19 21:58:04 +04:00
|
|
|
*/
|
2013-11-01 13:43:03 +01:00
|
|
|
public function __construct($rootAlias)
|
2012-08-19 21:58:04 +04:00
|
|
|
{
|
2013-11-01 13:43:03 +01:00
|
|
|
$this->rootAlias = $rootAlias;
|
2012-08-19 21:58:04 +04:00
|
|
|
$this->expr = new Expr();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-12-13 16:36:14 +00:00
|
|
|
* Gets bound parameters.
|
2012-08-19 21:58:04 +04:00
|
|
|
* Filled after {@link dispach()}.
|
|
|
|
*
|
|
|
|
* @return \Doctrine\Common\Collections\Collection
|
|
|
|
*/
|
|
|
|
public function getParameters()
|
|
|
|
{
|
2012-08-20 19:54:11 +04:00
|
|
|
return new ArrayCollection($this->parameters);
|
2012-08-19 21:58:04 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-12-13 16:36:14 +00:00
|
|
|
* Clears parameters.
|
|
|
|
*
|
|
|
|
* @return void
|
2012-08-19 21:58:04 +04:00
|
|
|
*/
|
|
|
|
public function clearParameters()
|
|
|
|
{
|
|
|
|
$this->parameters = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-12-13 16:36:14 +00:00
|
|
|
* Converts Criteria expression to Query one based on static map.
|
2012-08-19 21:58:04 +04:00
|
|
|
*
|
|
|
|
* @param string $criteriaOperator
|
|
|
|
*
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
private static function convertComparisonOperator($criteriaOperator)
|
|
|
|
{
|
|
|
|
return isset(self::$operatorMap[$criteriaOperator]) ? self::$operatorMap[$criteriaOperator] : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
|
|
|
public function walkCompositeExpression(CompositeExpression $expr)
|
|
|
|
{
|
|
|
|
$expressionList = array();
|
|
|
|
|
|
|
|
foreach ($expr->getExpressionList() as $child) {
|
|
|
|
$expressionList[] = $this->dispatch($child);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch($expr->getType()) {
|
|
|
|
case CompositeExpression::TYPE_AND:
|
|
|
|
return new Expr\Andx($expressionList);
|
|
|
|
|
|
|
|
case CompositeExpression::TYPE_OR:
|
|
|
|
return new Expr\Orx($expressionList);
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw new \RuntimeException("Unknown composite " . $expr->getType());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
|
|
|
public function walkComparison(Comparison $comparison)
|
|
|
|
{
|
|
|
|
$parameterName = str_replace('.', '_', $comparison->getField());
|
2013-11-19 11:09:23 -05:00
|
|
|
|
2013-11-18 10:36:46 -05:00
|
|
|
foreach($this->parameters as $parameter) {
|
|
|
|
if($parameter->getName() === $parameterName) {
|
|
|
|
$parameterName .= '_' . count($this->parameters);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-11-19 11:09:23 -05:00
|
|
|
|
2012-08-19 21:58:04 +04:00
|
|
|
$parameter = new Parameter($parameterName, $this->walkValue($comparison->getValue()));
|
|
|
|
$placeholder = ':' . $parameterName;
|
|
|
|
|
|
|
|
switch ($comparison->getOperator()) {
|
|
|
|
case Comparison::IN:
|
2012-08-20 19:54:11 +04:00
|
|
|
$this->parameters[] = $parameter;
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->in($this->rootAlias . '.' . $comparison->getField(), $placeholder);
|
2012-08-19 21:58:04 +04:00
|
|
|
|
|
|
|
case Comparison::NIN:
|
2012-08-20 19:54:11 +04:00
|
|
|
$this->parameters[] = $parameter;
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->notIn($this->rootAlias . '.' . $comparison->getField(), $placeholder);
|
2012-08-19 21:58:04 +04:00
|
|
|
|
|
|
|
case Comparison::EQ:
|
|
|
|
case Comparison::IS:
|
|
|
|
if ($this->walkValue($comparison->getValue()) === null) {
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->isNull($this->rootAlias . '.' . $comparison->getField());
|
2012-08-19 21:58:04 +04:00
|
|
|
}
|
2012-08-20 19:54:11 +04:00
|
|
|
$this->parameters[] = $parameter;
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->eq($this->rootAlias . '.' . $comparison->getField(), $placeholder);
|
2012-08-19 21:58:04 +04:00
|
|
|
|
|
|
|
case Comparison::NEQ:
|
|
|
|
if ($this->walkValue($comparison->getValue()) === null) {
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->isNotNull($this->rootAlias . '.' . $comparison->getField());
|
2012-08-19 21:58:04 +04:00
|
|
|
}
|
2012-08-20 19:54:11 +04:00
|
|
|
$this->parameters[] = $parameter;
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->neq($this->rootAlias . '.' . $comparison->getField(), $placeholder);
|
2012-08-19 21:58:04 +04:00
|
|
|
|
2013-09-16 14:56:04 +02:00
|
|
|
case Comparison::CONTAINS:
|
|
|
|
$parameter->setValue('%' . $parameter->getValue() . '%', $parameter->getType());
|
|
|
|
$this->parameters[] = $parameter;
|
2013-11-01 13:43:03 +01:00
|
|
|
return $this->expr->like($this->rootAlias . '.' . $comparison->getField(), $placeholder);
|
2013-09-16 14:56:04 +02:00
|
|
|
|
2012-08-19 21:58:04 +04:00
|
|
|
default:
|
|
|
|
$operator = self::convertComparisonOperator($comparison->getOperator());
|
|
|
|
if ($operator) {
|
2012-08-20 19:54:11 +04:00
|
|
|
$this->parameters[] = $parameter;
|
2012-08-19 21:58:04 +04:00
|
|
|
return new Expr\Comparison(
|
2013-11-01 13:43:03 +01:00
|
|
|
$this->rootAlias . '.' . $comparison->getField(),
|
2012-08-19 21:58:04 +04:00
|
|
|
$operator,
|
|
|
|
$placeholder
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new \RuntimeException("Unknown comparison operator: " . $comparison->getOperator());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
|
|
|
public function walkValue(Value $value)
|
|
|
|
{
|
|
|
|
return $value->getValue();
|
|
|
|
}
|
|
|
|
}
|