diff --git a/lib/Doctrine/ORM/Query/Parser.php b/lib/Doctrine/ORM/Query/Parser.php index 29fa3f18f..cd1269377 100644 --- a/lib/Doctrine/ORM/Query/Parser.php +++ b/lib/Doctrine/ORM/Query/Parser.php @@ -93,7 +93,14 @@ class Parser * @var EnityManager */ private $_em; - + + /** + * The Abtract Syntax Tree of processed DQL query. + * + * @var SelectStatement | UpdateStatement | DeleteStatement + */ + private $_ast; + /** * The Query to parse. * @@ -140,6 +147,7 @@ class Parser $this->_em = $query->getEntityManager(); $this->_lexer = new Lexer($query->getDql()); $this->_parserResult = new ParserResult(); + $this->_ast = null; } /** @@ -193,6 +201,16 @@ class Parser return $this->_em; } + /** + * Gets the Abstract Syntax Tree generated by the parser. + * + * @return SelectStatement | UpdateStatement | DeleteStatement + */ + public function getAST() + { + return $this->_ast; + } + /** * Registers a custom function that returns strings. * @@ -274,10 +292,8 @@ class Parser // Parse & build AST $AST = $this->QueryLanguage(); - // Check for end of string - if ($this->_lexer->lookahead !== null) { - $this->syntaxError('end of string'); - } + // Activate semantical checks after this point. Process all deferred checks in pipeline + $this->_processDeferredExpressionsStack($AST); if ($customWalkers = $this->_query->getHint(Query::HINT_CUSTOM_TREE_WALKERS)) { $this->_customTreeWalkers = $customWalkers; @@ -286,9 +302,11 @@ class Parser // Run any custom tree walkers over the AST if ($this->_customTreeWalkers) { $treeWalkerChain = new TreeWalkerChain($this->_query, $this->_parserResult, $this->_queryComponents); + foreach ($this->_customTreeWalkers as $walker) { $treeWalkerChain->addTreeWalker($walker); } + if ($AST instanceof AST\SelectStatement) { $treeWalkerChain->walkSelectStatement($AST); } else if ($AST instanceof AST\UpdateStatement) { @@ -492,11 +510,11 @@ class Parser * @param array $token * @param integer $nestingLevel */ - private function _subscribeExpression($expression, $method, $token, $nextingLevel = null) + private function _subscribeExpression($expression, $method, $token, $nestingLevel = null) { $nestingLevel = ($nestingLevel !== null) ?: $this->_nestingLevel; - $exprStack[] = array( + $exprStack = array( 'method' => $method, 'expression' => $expression, 'nestingLevel' => $nestingLevel, @@ -508,13 +526,17 @@ class Parser /** * Processes the topmost stack of deferred path expressions. + * + * @param mixed $AST */ - private function _processDeferredExpressionsStack() + private function _processDeferredExpressionsStack($AST) { foreach ($this->_deferredExpressionsStack as $item) { + if (!isset($item['method'])) var_dump($item); + $method = '_validate' . $item['method']; - $this->$method($item['expression'], $item['token'], $item['nestingLevel']); + $this->$method($item, $AST); } } @@ -523,30 +545,35 @@ class Parser * Validates that the given IdentificationVariable is a semantically correct. * It must exist in query components list. * - * @param string $identVariable - * @param array $token - * @param integer $nestingLevel + * @param array $deferredItem + * @param mixed $AST * * @return array Query Component */ - private function _validateIdentificationVariable($identVariable, $token, $nestingLevel) + private function _validateIdentificationVariable($deferredItem, $AST) { + $identVariable = $deferredItem['expression']; + // Check if IdentificationVariable exists in queryComponents if ( ! isset($this->_queryComponents[$identVariable])) { - $this->semanticalError("'$identVariable' is not defined.", $token); + $this->semanticalError( + "'$identVariable' is not defined.", $deferredItem['token'] + ); } $qComp = $this->_queryComponents[$identVariable]; // Check if queryComponent points to an AbstractSchemaName or a ResultVariable if ( ! isset($qComp['metadata'])) { - $this->semanticalError("'$identVariable' does not point to a Class.", $token); + $this->semanticalError( + "'$identVariable' does not point to a Class.", $deferredItem['token'] + ); } // Validate if identification variable nesting level is lower or equal than the current one - if ($qComp['nestingLevel'] > $nestingLevel) { + if ($qComp['nestingLevel'] > $deferredItem['nestingLevel']) { $this->semanticalError( - "'$identVariable' is used outside the scope of its declaration.", $token + "'$identVariable' is used outside the scope of its declaration.", $deferredItem['token'] ); } @@ -554,75 +581,38 @@ class Parser } /** - * Validates that the given AliasIdentificationVariable is a semantically correct. - * It must not exist in query components list. - * - * @param string $aliasIdentVariable - * @param array $token - * @param integer $nestingLevel - * - * @return boolean - */ - private function _validateAliasIdentificationVariable($aliasIdentVariable, $token, $nestingLevel) - { - $exists = isset($this->_queryComponents[$aliasIdentVariable]); - - if ($exists) { - $this->semanticalError("'$aliasIdentVariable' is already defined.", $token); - } - - return $exists; - } - - /** - * Validates that the given AbstractSchemaName is a semantically correct. - * It must be defined in the scope of Application. - * - * @param string $schemaName - * @param array $token - * @param integer $nestingLevel - * - * @return boolean - */ - private function _validateAbstractSchemaName($schemaName, $token, $nestingLevel) - { - $exists = class_exists($schemaName, true); - - if ( ! $exists) { - $this->semanticalError("Class '$schemaName' is not defined.", $token); - } - - return $exists; - } - - /** - * Validates that the given AliasIdentificationVariable is a semantically correct. + * Validates that the given ResultVariable is a semantically correct. * It must exist in query components list. * - * @param string $resultVariable - * @param array $token - * @param integer $nestingLevel + * @param array $deferredItem + * @param mixed $AST * * @return array Query Component */ - private function _validateResultVariable($resultVariable, $token, $nestingLevel) + private function _validateResultVariable($deferredItem, $AST) { + $resultVariable = $deferredItem['expression']; + // Check if ResultVariable exists in queryComponents if ( ! isset($this->_queryComponents[$resultVariable])) { - $this->semanticalError("'$resultVariable' is not defined.", $token); + $this->semanticalError( + "'$resultVariable' is not defined.", $deferredItem['token'] + ); } $qComp = $this->_queryComponents[$resultVariable]; // Check if queryComponent points to an AbstractSchemaName or a ResultVariable if ( ! isset($qComp['resultVariable'])) { - $this->semanticalError("'$identVariable' does not point to a ResultVariable.", $token); + $this->semanticalError( + "'$identVariable' does not point to a ResultVariable.", $deferredItem['token'] + ); } // Validate if identification variable nesting level is lower or equal than the current one - if ($qComp['nestingLevel'] > $nestingLevel) { + if ($qComp['nestingLevel'] > $deferredItem['nestingLevel']) { $this->semanticalError( - "'$resultVariable' is used outside the scope of its declaration.", $token + "'$resultVariable' is used outside the scope of its declaration.", $deferredItem['token'] ); } @@ -632,14 +622,14 @@ class Parser /** * Validates that the given JoinAssociationPathExpression is a semantically correct. * - * @param JoinAssociationPathExpression $pathExpression - * @param array $token - * @param integer $nestingLevel + * @param array $deferredItem + * @param mixed $AST * * @return array Query Component */ - private function _validateJoinAssociationPathExpression(AST\JoinAssociationPathExpression $pathExpression, $token, $nestingLevel) + private function _validateJoinAssociationPathExpression($deferredItem, $AST) { + $pathExpression = $deferredItem['expression']; $qComp = $this->_queryComponents[$pathExpression->identificationVariable];; // Validating association field (*-to-one or *-to-many) @@ -662,70 +652,106 @@ class Parser * SingleValuedAssociationPathExpression ::= IdentificationVariable "." {SingleValuedAssociationField "."}* SingleValuedAssociationField * CollectionValuedPathExpression ::= IdentificationVariable "." {SingleValuedAssociationField "."}* CollectionValuedAssociationField * - * @param PathExpression $pathExpression - * @param array $token - * @param integer $nestingLevel + * @param array $deferredItem + * @param mixed $AST * * @return integer */ - private function _validatePathExpression(AST\PathExpression $pathExpression, $token, $nestingLevel) + private function _validatePathExpression($deferredItem, $AST) { - $qComp = $this->_queryComponents[$pathExpression->identificationVariable]; + $pathExpression = $deferredItem['expression']; + $parts = $pathExpression->parts; + $numParts = count($parts); + $qComp = $this->_queryComponents[$pathExpression->identificationVariable]; + + $aliasIdentificationVariable = $pathExpression->identificationVariable; + $parentField = $pathExpression->identificationVariable; $class = $qComp['metadata']; - $stateField = $collectionField = null; - - foreach ($pathExpression->parts as $field) { + $fieldType = null; + $curIndex = 0; + + foreach ($parts as $field) { // Check if it is not in a state field - if ($stateField !== null) { + if ($fieldType & AST\PathExpression::TYPE_STATE_FIELD) { $this->semanticalError( - 'Cannot navigate through state field named ' . $stateField, $token + 'Cannot navigate through state field named ' . $field . ' on ' . $parentField, + $deferredItem['token'] ); } // Check if it is not a collection field - if ($collectionField !== null) { + if ($fieldType & AST\PathExpression::TYPE_COLLECTION_VALUED_ASSOCIATION) { $this->semanticalError( - 'Cannot navigate through collection-valued field named ' . $collectionField, - $token + 'Cannot navigate through collection field named ' . $field . ' on ' . $parentField, + $deferredItem['token'] ); } - // Check if field exists + // Check if field or association exists if ( ! isset($class->associationMappings[$field]) && ! isset($class->fieldMappings[$field])) { $this->semanticalError( - 'Class ' . $class->name . ' has no field or association named ' . $field, $token + 'Class ' . $class->name . ' has no field or association named ' . $field, + $deferredItem['token'] ); } + + $parentField = $field; if (isset($class->fieldMappings[$field])) { - $stateField = $field; - } else if ($class->associationMappings[$field]->isOneToOne()) { - $class = $this->_em->getClassMetadata($class->associationMappings[$field]->targetEntityName); + $fieldType = AST\PathExpression::TYPE_STATE_FIELD; } else { - $collectionField = $field; + $assoc = $class->associationMappings[$field]; + $class = $this->_em->getClassMetadata($assoc->targetEntityName); + + if ( + ($curIndex != $numParts - 1) && + ! isset($this->_queryComponents[$aliasIdentificationVariable . '.' . $field]) + ) { + // Building queryComponent + $joinQueryComponent = array( + 'metadata' => $class, + 'parent' => $aliasIdentificationVariable, + 'relation' => $assoc, + 'map' => null, + 'nestingLevel' => $this->_nestingLevel, + 'token' => $deferredItem['token'], + ); + + // Create AST node + $joinVariableDeclaration = new AST\JoinVariableDeclaration( + new AST\Join( + AST\Join::JOIN_TYPE_INNER, + new AST\JoinAssociationPathExpression($aliasIdentificationVariable, $field), + $aliasIdentificationVariable . '.' . $field + ), + null + ); + $AST->fromClause->identificationVariableDeclarations[0]->joinVariableDeclarations[] = $joinVariableDeclaration; + + $this->_queryComponents[$aliasIdentificationVariable . '.' . $field] = $joinQueryComponent; + } + + $aliasIdentificationVariable .= '.' . $field; + + if ($assoc->isOneToOne()) { + $fieldType = AST\PathExpression::TYPE_SINGLE_VALUED_ASSOCIATION; + } else { + $fieldType = AST\PathExpression::TYPE_COLLECTION_VALUED_ASSOCIATION; + } } + + $curIndex++; } - // Recognize correct expression type - $expressionType = null; - - if ($stateField !== null) { - $expressionType = AST\PathExpression::TYPE_STATE_FIELD; - } else if ($collectionField !== null) { - $expressionType = AST\PathExpression::TYPE_COLLECTION_VALUED_ASSOCIATION; - } else { - $expressionType = AST\PathExpression::TYPE_SINGLE_VALUED_ASSOCIATION; - } - // Validate if PathExpression is one of the expected types $expectedType = $pathExpression->expectedType; - if ( ! ($expectedType & $expressionType)) { + if ( ! ($expectedType & $fieldType)) { // We need to recognize which was expected type(s) $expectedStringTypes = array(); - // Validate state field type (field/column) + // Validate state field type if ($expectedType & AST\PathExpression::TYPE_STATE_FIELD) { $expectedStringTypes[] = 'StateFieldPathExpression'; } @@ -749,15 +775,14 @@ class Parser $semanticalError .= implode(' or ', $expectedStringTypes) . ' expected.'; } - $this->semanticalError($semanticalError, $token); + $this->semanticalError($semanticalError, $deferredItem['token']); } // We need to force the type in PathExpression - $pathExpression->type = $expressionType; + $pathExpression->type = $fieldType; - return $expressionType; + return $fieldType; } - /** * QueryLanguage ::= SelectStatement | UpdateStatement | DeleteStatement @@ -769,7 +794,7 @@ class Parser public function QueryLanguage() { $this->_lexer->moveNext(); - + switch ($this->_lexer->lookahead['type']) { case Lexer::T_SELECT: return $this->SelectStatement(); @@ -784,6 +809,11 @@ class Parser $this->syntaxError('SELECT, UPDATE or DELETE'); break; } + + // Check for end of string + if ($this->_lexer->lookahead !== null) { + $this->syntaxError('end of string'); + } } @@ -807,10 +837,7 @@ class Parser $selectStatement->orderByClause = $this->_lexer->isNextToken(Lexer::T_ORDER) ? $this->OrderByClause() : null; - - // Activate semantical checks after this point. Process all deferred checks in pipeline - $this->_processDeferredExpressionsStack(); - + return $selectStatement; } @@ -824,9 +851,6 @@ class Parser $updateStatement = new AST\UpdateStatement($this->UpdateClause()); $updateStatement->whereClause = $this->_lexer->isNextToken(Lexer::T_WHERE) ? $this->WhereClause() : null; - - // Activate semantical checks after this point. Process all deferred checks in pipeline - $this->_processDeferredExpressionsStack(); return $updateStatement; } @@ -841,9 +865,6 @@ class Parser $deleteStatement = new AST\DeleteStatement($this->DeleteClause()); $deleteStatement->whereClause = $this->_lexer->isNextToken(Lexer::T_WHERE) ? $this->WhereClause() : null; - - // Activate semantical checks after this point. Process all deferred checks in pipeline - $this->_processDeferredExpressionsStack(); return $deleteStatement; } @@ -883,11 +904,13 @@ class Parser $this->match(Lexer::T_IDENTIFIER); $aliasIdentVariable = $this->_lexer->token['value']; + $exists = isset($this->_queryComponents[$aliasIdentVariable]); - // Apply AliasIdentificationVariable validation - $this->_validateAliasIdentificationVariable( - $aliasIdentVariable, $this->_lexer->token, $this->_nestingLevel - ); + if ($exists) { + $this->semanticalError( + "'$aliasIdentVariable' is already defined.", $this->_lexer->token + ); + } return $aliasIdentVariable; } @@ -902,15 +925,36 @@ class Parser $this->match(Lexer::T_IDENTIFIER); $schemaName = $this->_lexer->token['value']; + $exists = class_exists($schemaName, true); - // Apply AbstractSchemaName validation - $this->_validateAbstractSchemaName( - $schemaName, $this->_lexer->token, $this->_nestingLevel - ); + if ( ! $exists) { + $this->semanticalError("Class '$schemaName' is not defined.", $this->_lexer->token); + } return $schemaName; } + /** + * AliasResultVariable ::= identifier + * + * @return string + */ + public function AliasResultVariable() + { + $this->match(Lexer::T_IDENTIFIER); + + $resultVariable = $this->_lexer->token['value']; + $exists = isset($this->_queryComponents[$resultVariable]); + + if ($exists) { + $this->semanticalError( + "'$resultVariable' is already defined.", $this->_lexer->token + ); + } + + return $resultVariable; + } + /** * ResultVariable ::= identifier * @@ -923,15 +967,11 @@ class Parser $resultVariable = $this->_lexer->token['value']; // Defer ResultVariable validation - $exprStack = array( - 'method' => 'ResultVariable', - 'expression' => $resultVariable, - 'nestingLevel' => $this->_nestingLevel, - 'token' => $this->_lexer->token, + $this->_subscribeExpression( + $resultVariable, 'ResultVariable', + $this->_lexer->token, $this->_nestingLevel ); - - array_push($this->_deferredExpressionsStack, $exprStack); - + return $resultVariable; } @@ -953,14 +993,10 @@ class Parser $pathExpr = new AST\JoinAssociationPathExpression($identVariable, $field); // Defer JoinAssociationPathExpression validation - $exprStack = array( - 'method' => 'JoinAssociationPathExpression', - 'expression' => $pathExpr, - 'nestingLevel' => $this->_nestingLevel, - 'token' => $token, + $this->_subscribeExpression( + $pathExpr, 'JoinAssociationPathExpression', + $token, $this->_nestingLevel ); - - array_push($this->_deferredExpressionsStack, $exprStack); return $pathExpr; } @@ -969,7 +1005,7 @@ class Parser * Parses an arbitrary path expression and defers semantical validation * based on expected types. * - * PathExpression ::= IdentificationVariable "." {identifier "."}* identifier + * PathExpression ::= IdentificationVariable {"." identifier}* "." identifier * * @param integer $expectedTypes * @return \Doctrine\ORM\Query\AST\PathExpression @@ -991,14 +1027,10 @@ class Parser $pathExpr = new AST\PathExpression($expectedTypes, $identVariable, $parts); // Defer PathExpression validation if requested to be defered - $exprStack = array( - 'method' => 'PathExpression', - 'expression' => $pathExpr, - 'nestingLevel' => $this->_nestingLevel, - 'token' => $token, + $this->_subscribeExpression( + $pathExpr, 'PathExpression', + $token, $this->_nestingLevel ); - - array_push($this->_deferredExpressionsStack, $exprStack); return $pathExpr; } @@ -1598,7 +1630,7 @@ class Parser ); } - $targetClassName = $parentClass->getAssociationMapping($assocField)->getTargetEntityName(); + $targetClassName = $parentClass->getAssociationMapping($assocField)->targetEntityName; // Building queryComponent $joinQueryComponent = array( @@ -1651,7 +1683,7 @@ class Parser /** * SelectExpression ::= * IdentificationVariable | StateFieldPathExpression | - * (AggregateExpression | "(" Subselect ")" | FunctionDeclaration) [["AS"] ResultVariable] + * (AggregateExpression | "(" Subselect ")" | FunctionDeclaration) [["AS"] AliasResultVariable] * * @return \Doctrine\ORM\Query\AST\SelectExpression */ @@ -1683,9 +1715,9 @@ class Parser if ($this->_lexer->isNextToken(Lexer::T_IDENTIFIER)) { $token = $this->_lexer->lookahead; - $fieldAliasIdentificationVariable = $this->ResultVariable(); + $fieldAliasIdentificationVariable = $this->AliasResultVariable(); - // Include ResultVariable in query components. + // Include AliasResultVariable in query components. $this->_queryComponents[$fieldAliasIdentificationVariable] = array( 'resultVariable' => $expression, 'nestingLevel' => $this->_nestingLevel, @@ -1708,7 +1740,7 @@ class Parser } /** - * SimpleSelectExpression ::= StateFieldPathExpression | IdentificationVariable | (AggregateExpression [["AS"] ResultVariable]) + * SimpleSelectExpression ::= StateFieldPathExpression | IdentificationVariable | (AggregateExpression [["AS"] AliasResultVariable]) * * @return \Doctrine\ORM\Query\AST\SimpleSelectExpression */ @@ -1735,10 +1767,10 @@ class Parser if ($this->_lexer->isNextToken(Lexer::T_IDENTIFIER)) { $token = $this->_lexer->lookahead; - $resultVariable = $this->ResultVariable(); + $resultVariable = $this->AliasResultVariable(); $expr->fieldIdentificationVariable = $resultVariable; - // Include ResultVariable in query components. + // Include AliasResultVariable in query components. $this->_queryComponents[$resultVariable] = array( 'resultvariable' => $expr, 'nestingLevel' => $this->_nestingLevel, diff --git a/lib/Doctrine/ORM/Query/SqlWalker.php b/lib/Doctrine/ORM/Query/SqlWalker.php index 83400e5ad..4c3bd80b5 100644 --- a/lib/Doctrine/ORM/Query/SqlWalker.php +++ b/lib/Doctrine/ORM/Query/SqlWalker.php @@ -414,9 +414,8 @@ class SqlWalker implements TreeWalker switch ($pathExpr->type) { case AST\PathExpression::TYPE_STATE_FIELD: $parts = $pathExpr->parts; - $numParts = count($parts); - $dqlAlias = $pathExpr->identificationVariable; - $fieldName = $parts[$numParts - 1]; + $fieldName = array_pop($parts); + $dqlAlias = $pathExpr->identificationVariable . (( ! empty($parts)) ? '.' . implode('.', $parts) : ''); $qComp = $this->_queryComponents[$dqlAlias]; $class = $qComp['metadata']; @@ -609,20 +608,18 @@ class SqlWalker implements TreeWalker */ public function walkOrderByItem($orderByItem) { + $sql = ''; $expr = $orderByItem->expression; + if ($expr instanceof AST\PathExpression) { - $parts = $expr->parts; - $dqlAlias = $expr->identificationVariable; - $class = $this->_queryComponents[$dqlAlias]['metadata']; - $columnName = $class->getQuotedColumnName($parts[0], $this->_platform); - - return $this->getSqlTableAlias($class->getTableName(), $dqlAlias) . '.' - . $columnName . ' ' . strtoupper($orderByItem->type); + $sql = $this->walkPathExpression($expr); } else { $columnName = $this->_queryComponents[$expr]['token']['value']; - return $this->_scalarResultAliasMap[$columnName] . ' ' . strtoupper($orderByItem->type); + $sql = $this->_scalarResultAliasMap[$columnName]; } + + return $sql . ' ' . strtoupper($orderByItem->type);; } /** @@ -771,16 +768,15 @@ class SqlWalker implements TreeWalker if ($expr instanceof AST\PathExpression) { if ($expr->type == AST\PathExpression::TYPE_STATE_FIELD) { $parts = $expr->parts; - $numParts = count($parts); - $dqlAlias = $expr->identificationVariable; - $fieldName = $parts[$numParts - 1]; + $fieldName = array_pop($parts); + $dqlAlias = $expr->identificationVariable . (( ! empty($parts)) ? '.' . implode('.', $parts) : ''); $qComp = $this->_queryComponents[$dqlAlias]; $class = $qComp['metadata']; if ( ! isset($this->_selectedClasses[$dqlAlias])) { $this->_selectedClasses[$dqlAlias] = $class; } - + $sqlTableAlias = $this->getSqlTableAlias($class->getTableName(), $dqlAlias); $columnName = $class->getQuotedColumnName($fieldName, $this->_platform); $columnAlias = $this->getSqlColumnAlias($class->columnNames[$fieldName]); @@ -1004,17 +1000,8 @@ class SqlWalker implements TreeWalker */ public function walkAggregateExpression($aggExpression) { - $sql = ''; - $parts = $aggExpression->pathExpression->parts; - $dqlAlias = $aggExpression->pathExpression->identificationVariable; - $fieldName = $parts[0]; - - $qComp = $this->_queryComponents[$dqlAlias]; - $columnName = $qComp['metadata']->getQuotedColumnName($fieldName, $this->_platform); - return $aggExpression->functionName . '(' . ($aggExpression->isDistinct ? 'DISTINCT ' : '') - . $this->getSqlTableAlias($qComp['metadata']->getTableName(), $dqlAlias) . '.' - . $columnName . ')'; + . $this->walkPathExpression($aggExpression->pathExpression) . ')'; } /** @@ -1038,12 +1025,7 @@ class SqlWalker implements TreeWalker */ public function walkGroupByItem(AST\PathExpression $pathExpr) { - $parts = $pathExpr->parts; - $dqlAlias = $pathExpr->identificationVariable; - $qComp = $this->_queryComponents[$dqlAlias]; - $columnName = $qComp['metadata']->getQuotedColumnName($parts[0], $this->_platform); - - return $this->getSqlTableAlias($qComp['metadata']->getTableName(), $dqlAlias) . '.' . $columnName; + return $this->walkPathExpression($pathExpr); } /** @@ -1219,8 +1201,10 @@ class SqlWalker implements TreeWalker $sql .= 'EXISTS (SELECT 1 FROM '; $entityExpr = $collMemberExpr->entityExpression; $collPathExpr = $collMemberExpr->collectionValuedPathExpression; + $parts = $collPathExpr->parts; - $dqlAlias = $collPathExpr->identificationVariable; + $fieldName = array_pop($parts); + $dqlAlias = $collPathExpr->identificationVariable . (( ! empty($parts)) ? '.' . implode('.', $parts) : ''); $class = $this->_queryComponents[$dqlAlias]['metadata']; @@ -1232,7 +1216,7 @@ class SqlWalker implements TreeWalker throw DoctrineException::notImplemented(); } - $assoc = $class->associationMappings[$parts[0]]; + $assoc = $class->associationMappings[$fieldName]; if ($assoc->isOneToMany()) { $targetClass = $this->_em->getClassMetadata($assoc->targetEntityName); diff --git a/tests/Doctrine/Tests/ORM/Query/LanguageRecognitionTest.php b/tests/Doctrine/Tests/ORM/Query/LanguageRecognitionTest.php index dd29657cf..adedff266 100644 --- a/tests/Doctrine/Tests/ORM/Query/LanguageRecognitionTest.php +++ b/tests/Doctrine/Tests/ORM/Query/LanguageRecognitionTest.php @@ -58,6 +58,25 @@ class LanguageRecognitionTest extends \Doctrine\Tests\OrmTestCase return $parser->parse(); } + + public function parseDqlForAST($dql, $hints = array()) + { + $query = $this->_em->createQuery($dql); + $query->setHint(Query::HINT_FORCE_PARTIAL_LOAD, true); + $query->setDql($dql); + + foreach ($hints as $key => $value) { + $query->setHint($key, $value); + } + + $parser = new \Doctrine\ORM\Query\Parser($query); + // We do NOT test SQL output here. That only unnecessarily slows down the tests! + $parser->setCustomOutputTreeWalker('Doctrine\Tests\Mocks\MockTreeWalker'); + $parser->parse(); + + return $parser->getAST(); + } + public function testEmptyQueryString() { @@ -84,6 +103,11 @@ class LanguageRecognitionTest extends \Doctrine\Tests\OrmTestCase $this->assertValidDql('SELECT u, p FROM Doctrine\Tests\Models\CMS\CmsUser u JOIN u.phonenumbers p'); } + public function testSelectMultipleComponentsWithAsterisk2() + { + $this->assertValidDql('SELECT a.user.name FROM Doctrine\Tests\Models\CMS\CmsArticle a'); + } + public function testSelectDistinctIsSupported() { $this->assertValidDql('SELECT DISTINCT u.name FROM Doctrine\Tests\Models\CMS\CmsUser u'); @@ -93,6 +117,11 @@ class LanguageRecognitionTest extends \Doctrine\Tests\OrmTestCase { $this->assertValidDql('SELECT COUNT(u.id) FROM Doctrine\Tests\Models\CMS\CmsUser u'); } + + public function testDuplicatedAliasInAggregateFunction() + { + $this->assertInvalidDql('SELECT COUNT(u.id) AS num, SUM(u.id) AS num FROM Doctrine\Tests\Models\CMS\CmsUser u'); + } public function testAggregateFunctionWithDistinctInSelect() { diff --git a/tests/Doctrine/Tests/ORM/Query/SelectSqlGenerationTest.php b/tests/Doctrine/Tests/ORM/Query/SelectSqlGenerationTest.php index 2674382cf..3fbb31ec6 100644 --- a/tests/Doctrine/Tests/ORM/Query/SelectSqlGenerationTest.php +++ b/tests/Doctrine/Tests/ORM/Query/SelectSqlGenerationTest.php @@ -2,7 +2,8 @@ namespace Doctrine\Tests\ORM\Query; -use Doctrine\ORM\Query; +use Doctrine\ORM\Query, + Doctrine\Common\DoctrineException; require_once __DIR__ . '/../../TestInit.php'; @@ -22,7 +23,7 @@ class SelectSqlGenerationTest extends \Doctrine\Tests\OrmTestCase $query->setHint(Query::HINT_FORCE_PARTIAL_LOAD, true); parent::assertEquals($sqlToBeConfirmed, $query->getSql()); $query->free(); - } catch (Doctrine_Exception $e) { + } catch (DoctrineException $e) { $this->fail($e->getMessage()); } } @@ -42,6 +43,22 @@ class SelectSqlGenerationTest extends \Doctrine\Tests\OrmTestCase 'SELECT c0_.id AS id0 FROM cms_users c0_' ); } + + public function testSupportsSelectForOneNestedField() + { + $this->assertSqlGeneration( + 'SELECT a.user.id FROM Doctrine\Tests\Models\CMS\CmsArticle a', + 'SELECT c0_.id AS id0 FROM cms_articles c1_ INNER JOIN cms_users c0_ ON c1_.user_id = c0_.id' + ); + } + + public function testSupportsSelectForAllNestedField() + { + $this->assertSqlGeneration( + 'SELECT a FROM Doctrine\Tests\Models\CMS\CmsArticle a ORDER BY a.user.name ASC', + 'SELECT c0_.id AS id0, c0_.topic AS topic1, c0_.text AS text2 FROM cms_articles c0_ INNER JOIN cms_users c1_ ON c0_.user_id = c1_.id ORDER BY c1_.name ASC' + ); + } public function testSupportsSelectForMultipleColumnsOfASingleComponent() {