Skip to content

Commit

Permalink
Merge branch 'DBAL-172'
Browse files Browse the repository at this point in the history
  • Loading branch information
beberlei committed Oct 30, 2011
2 parents cf93b0d + dd79627 commit 08e6c48
Show file tree
Hide file tree
Showing 3 changed files with 234 additions and 166 deletions.
77 changes: 42 additions & 35 deletions lib/Doctrine/DBAL/Query/QueryBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,10 @@

/**
* QueryBuilder class is responsible to dynamically create SQL queries.
*
*
* Important: Verify that every feature you use will work with your database vendor.
* SQL Query Builder does not attempt to validate the generated SQL at all.
*
*
* The query builder does no validation whatsoever if certain features even work with the
* underlying database vendor. Limit queries and joins are NOT applied to UPDATE and DELETE statements
* even if some vendors such as MySQL support it.
Expand Down Expand Up @@ -102,10 +102,10 @@ class QueryBuilder
* @var integer The maximum number of results to retrieve.
*/
private $maxResults = null;

/**
* The counter of bound parameters used with {@see bindValue)
*
*
* @var int
*/
private $boundCounter = 0;
Expand Down Expand Up @@ -170,14 +170,14 @@ public function getState()
{
return $this->state;
}

/**
* Execute this query using the bound parameters and their types.
*
*
* Uses {@see Connection::executeQuery} for select statements and {@see Connection::executeUpdate}
* for insert, update and delete statements.
*
* @return mixed
*
* @return mixed
*/
public function execute()
{
Expand Down Expand Up @@ -390,7 +390,7 @@ public function add($sqlPartName, $sqlPart, $append = false)
}

$this->sqlParts[$sqlPartName] = $sqlPart;

return $this;
}

Expand Down Expand Up @@ -604,7 +604,7 @@ public function leftJoin($fromAlias, $join, $alias, $condition = null)
)
), true);
}

/**
* Creates and adds a right join to the query.
*
Expand Down Expand Up @@ -939,69 +939,76 @@ public function resetQueryPart($queryPartName)

return $this;
}

/**
* Converts this instance into a SELECT string in SQL.
*
*
* @return string
*/
private function getSQLForSelect()
{
$query = 'SELECT ' . implode(', ', $this->sqlParts['select']) . ' FROM ';

$fromClauses = array();

// Loop through all FROM clauses
foreach ($this->sqlParts['from'] as $from) {
$fromClause = $from['table'] . ' ' . $from['alias'];

if (isset($this->sqlParts['join'][$from['alias']])) {
foreach ($this->sqlParts['join'][$from['alias']] as $join) {
$fromClause .= ' ' . strtoupper($join['joinType'])
. ' JOIN ' . $join['joinTable'] . ' ' . $join['joinAlias']
$fromClause .= ' ' . strtoupper($join['joinType'])
. ' JOIN ' . $join['joinTable'] . ' ' . $join['joinAlias']
. ' ON ' . ((string) $join['joinCondition']);
}
}
$fromClauses[] = $fromClause;

$fromClauses[$from['alias']] = $fromClause;
}

$query .= implode(', ', $fromClauses)

// loop through all JOIN clasues for validation purpose
foreach ($this->sqlParts['join'] as $fromAlias => $joins) {
if ( ! isset($fromClauses[$fromAlias]) ) {
throw QueryException::unknownFromAlias($fromAlias, array_keys($fromClauses));
}
}

$query .= implode(', ', $fromClauses)
. ($this->sqlParts['where'] !== null ? ' WHERE ' . ((string) $this->sqlParts['where']) : '')
. ($this->sqlParts['groupBy'] ? ' GROUP BY ' . implode(', ', $this->sqlParts['groupBy']) : '')
. ($this->sqlParts['having'] !== null ? ' HAVING ' . ((string) $this->sqlParts['having']) : '')
. ($this->sqlParts['orderBy'] ? ' ORDER BY ' . implode(', ', $this->sqlParts['orderBy']) : '');
return ($this->maxResults === null && $this->firstResult == null)

return ($this->maxResults === null && $this->firstResult == null)
? $query
: $this->connection->getDatabasePlatform()->modifyLimitQuery($query, $this->maxResults, $this->firstResult);
}

/**
* Converts this instance into an UPDATE string in SQL.
*
*
* @return string
*/
private function getSQLForUpdate()
{
$table = $this->sqlParts['from']['table'] . ($this->sqlParts['from']['alias'] ? ' ' . $this->sqlParts['from']['alias'] : '');
$query = 'UPDATE ' . $table
$query = 'UPDATE ' . $table
. ' SET ' . implode(", ", $this->sqlParts['set'])
. ($this->sqlParts['where'] !== null ? ' WHERE ' . ((string) $this->sqlParts['where']) : '');

return $query;
}

/**
* Converts this instance into a DELETE string in SQL.
*
*
* @return string
*/
private function getSQLForDelete()
{
$table = $this->sqlParts['from']['table'] . ($this->sqlParts['from']['alias'] ? ' ' . $this->sqlParts['from']['alias'] : '');
$query = 'DELETE FROM ' . $table . ($this->sqlParts['where'] !== null ? ' WHERE ' . ((string) $this->sqlParts['where']) : '');

return $query;
}

Expand All @@ -1015,7 +1022,7 @@ public function __toString()
{
return $this->getSQL();
}

/**
* Create a new named parameter and bind the value $value to it.
*
Expand Down Expand Up @@ -1053,15 +1060,15 @@ public function createNamedParameter( $value, $type = \PDO::PARAM_STR, $placeHol

return $placeHolder;
}

/**
* Create a new positional parameter and bind the given value to it.
*
*
* Attention: If you are using positional parameters with the query builder you have
* to be very careful to bind all parameters in the order they appear in the SQL
* statement , otherwise they get bound in the wrong order which can lead to serious
* bugs in your code.
*
*
* Example:
* <code>
* $qb = $conn->createQueryBuilder();
Expand All @@ -1070,7 +1077,7 @@ public function createNamedParameter( $value, $type = \PDO::PARAM_STR, $placeHol
* ->where('u.username = ' . $qb->createPositionalParameter('Foo', PDO::PARAM_STR))
* ->orWhere('u.username = ' . $qb->createPositionalParameter('Bar', PDO::PARAM_STR))
* </code>
*
*
* @param mixed $value
* @param mixed $type
* @return string
Expand Down
40 changes: 40 additions & 0 deletions lib/Doctrine/DBAL/Query/QueryException.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
<?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 LGPL. For more information, see
* <http://www.doctrine-project.org>.
*/

namespace Doctrine\DBAL\Query;

use Doctrine\DBAL\DBALException;

/**
* Driver interface.
* Interface that all DBAL drivers must implement.
*
* @since 2.1.4
*/
class QueryException extends DBALException
{
static public function unknownFromAlias($alias, $registeredAliases)
{
return new self("The given alias '" . $alias . "' is not part of " .
"any FROM clause table. The currently registered FROM-clause " .
"aliases are: " . implode(", ", $registeredAliases) . ". Join clauses " .
"are bound to from clauses to provide support for mixing of multiple " .
"from and join clauses.");
}
}
Loading

0 comments on commit 08e6c48

Please sign in to comment.