HHH-9090 : HQL parser is trying to reuse parent implied join for subquery

(cherry picked from commit 153c4e32ef)
This commit is contained in:
Gail Badner 2014-08-12 16:11:54 -07:00
parent a1b62a63c7
commit 57ac3f0b2e
2 changed files with 89 additions and 24 deletions

View File

@ -476,7 +476,7 @@ public class DotNode extends FromReferenceNode implements DisplayableNode, Selec
boolean found = elem != null; boolean found = elem != null;
// even though we might find a pre-existing element by join path, we may not be able to reuse it... // even though we might find a pre-existing element by join path, we may not be able to reuse it...
boolean useFoundFromElement = found && canReuse( elem, classAlias ); boolean useFoundFromElement = found && canReuse( elem );
if ( !useFoundFromElement ) { if ( !useFoundFromElement ) {
// If this is an implied join in a from element, then use the impled join type which is part of the // If this is an implied join in a from element, then use the impled join type which is part of the
@ -525,12 +525,7 @@ public class DotNode extends FromReferenceNode implements DisplayableNode, Selec
setFromElement( elem ); // This 'dot' expression now refers to the resulting from element. setFromElement( elem ); // This 'dot' expression now refers to the resulting from element.
} }
private boolean canReuse(FromElement fromElement, String requestedAlias) { private boolean canReuse(FromElement fromElement) {
// implicit joins are always(?) ok to reuse
if ( isImplicitJoin( fromElement ) ) {
return true;
}
// if the from-clauses are the same, we can be a little more aggressive in terms of what we reuse // if the from-clauses are the same, we can be a little more aggressive in terms of what we reuse
if ( fromElement.getFromClause() == getWalker().getCurrentFromClause() ) { if ( fromElement.getFromClause() == getWalker().getCurrentFromClause() ) {
return true; return true;
@ -540,10 +535,6 @@ public class DotNode extends FromReferenceNode implements DisplayableNode, Selec
return getWalker().getCurrentClauseType() != SqlTokenTypes.FROM; return getWalker().getCurrentClauseType() != SqlTokenTypes.FROM;
} }
private boolean isImplicitJoin(FromElement fromElement) {
return fromElement.isImplied();
}
private void setImpliedJoin(FromElement elem) { private void setImpliedJoin(FromElement elem) {
this.impliedJoin = elem; this.impliedJoin = elem;
if ( getFirstChild().getType() == SqlTokenTypes.DOT ) { if ( getFirstChild().getType() == SqlTokenTypes.DOT ) {

View File

@ -23,22 +23,24 @@
*/ */
package org.hibernate.test.hql; package org.hibernate.test.hql;
import java.util.Set; import java.util.ArrayList;
import java.util.List;
import javax.persistence.Entity; import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id; import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.OneToMany; import javax.persistence.OneToMany;
import javax.persistence.OneToOne; import javax.persistence.OneToOne;
import javax.persistence.Table; import javax.persistence.Table;
import org.hibernate.Session; import org.hibernate.Session;
import org.hibernate.testing.FailureExpected;
import org.hibernate.testing.TestForIssue; import org.hibernate.testing.TestForIssue;
import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase; import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
import org.junit.Test; import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull;
/** /**
* NOTE : some subquery related tests still exist in other test classes in the suite. This is a later * NOTE : some subquery related tests still exist in other test classes in the suite. This is a later
* attempt to create a more targeted set of subquery related tests. * attempt to create a more targeted set of subquery related tests.
@ -51,6 +53,7 @@ public class SubQueryTest extends BaseCoreFunctionalTestCase {
@Table( name = "ROOT" ) @Table( name = "ROOT" )
public static class Root { public static class Root {
@Id @Id
@GeneratedValue
public Integer id; public Integer id;
public String rootName; public String rootName;
@OneToOne @OneToOne
@ -62,17 +65,19 @@ public class SubQueryTest extends BaseCoreFunctionalTestCase {
@Table( name = "BRANCH" ) @Table( name = "BRANCH" )
public static class Branch { public static class Branch {
@Id @Id
@GeneratedValue
public Integer id; public Integer id;
public String branchName; public String branchName;
@OneToMany @OneToMany
public Set<Leaf> leaves; public List<Leaf> leaves;
} }
@Entity( name = "Leaf" ) @Entity( name = "Leaf" )
@Table( name = "LEAF" ) @Table( name = "LEAF" )
public static class Leaf { public static class Leaf {
@Id @Id
@GeneratedValue
public Integer id; public Integer id;
public String leafName; public String leafName;
} }
@ -84,18 +89,87 @@ public class SubQueryTest extends BaseCoreFunctionalTestCase {
@Test @Test
@TestForIssue( jiraKey = "HHH-9090" ) @TestForIssue( jiraKey = "HHH-9090" )
@FailureExpected( jiraKey = "HHH-9090" )
public void testCorrelatedJoin() { public void testCorrelatedJoin() {
Session s = openSession(); Session s = openSession();
s.beginTransaction(); s.beginTransaction();
Root root = new Root();
// simple syntax check of the generated SQL root.rootName = "root name";
final String qry = "from Root as r " + root.branch = new Branch();
"where r.branch.branchName = 'some branch name' " + root.branch.branchName = "branch";
" and exists( from r.branch.leaves as s where s.leafName = 'some leaf name')"; root.branch.leaves = new ArrayList<Leaf>();
s.createQuery( qry ).list(); Leaf leaf1 = new Leaf();
leaf1.leafName = "leaf1";
Leaf leaf2 = new Leaf();
leaf2.leafName = "leaf2";
root.branch.leaves.add( leaf1 );
root.branch.leaves.add( leaf2 );
s.persist( leaf1 );
s.persist( leaf2 );
s.persist( root.branch );
s.persist( root );
Root otherRoot = new Root();
otherRoot.rootName = "other root name";
otherRoot.branch = new Branch();
otherRoot.branch.branchName = "other branch";
otherRoot.branch.leaves = new ArrayList<Leaf>();
Leaf otherLeaf1 = new Leaf();
otherLeaf1.leafName = "leaf1";
Leaf otherLeaf3 = new Leaf();
otherLeaf3.leafName = "leaf3";
otherRoot.branch.leaves.add( otherLeaf1 );
otherRoot.branch.leaves.add( otherLeaf3 );
s.persist( otherLeaf1 );
s.persist( otherLeaf3 );
s.persist( otherRoot.branch );
s.persist( otherRoot );
s.getTransaction().commit(); s.getTransaction().commit();
s.close(); s.close();
s = openSession();
s.beginTransaction();
String qry = "from Root as r " +
"where r.branch.branchName = 'branch' " +
" and exists( from r.branch.leaves as s where s.leafName = 'leaf1')";
Root rootQueried = (Root) s.createQuery( qry ).uniqueResult();
assertEquals( root.rootName, rootQueried.rootName );
assertEquals( root.branch.branchName, rootQueried.branch.branchName );
assertEquals( leaf1.leafName, rootQueried.branch.leaves.get( 0 ).leafName );
assertEquals( leaf2.leafName, rootQueried.branch.leaves.get( 1 ).leafName );
s.getTransaction().commit();
s.close();
s = openSession();
s.beginTransaction();
qry = "from Root as r " +
"where r.branch.branchName = 'branch' " +
" and exists( from r.branch.leaves as s where s.leafName = 'leaf3')";
assertNull( s.createQuery( qry ).uniqueResult() );
s.getTransaction().commit();
s.close();
s = openSession();
s.beginTransaction();
qry = "from Root as r " +
"where exists( from r.branch.leaves as s where r.branch.branchName = 'branch' and s.leafName = 'leaf1')";
rootQueried = (Root) s.createQuery( qry ).uniqueResult();
assertEquals( root.rootName, rootQueried.rootName );
assertEquals( root.branch.branchName, rootQueried.branch.branchName );
assertEquals( leaf1.leafName, rootQueried.branch.leaves.get( 0 ).leafName );
assertEquals( leaf2.leafName, rootQueried.branch.leaves.get( 1 ).leafName );
s.getTransaction().commit();
s.close();
s = openSession();
s.beginTransaction();
qry = "from Root as r" +
" where exists( from Root r1 where r1.branch.branchName = r.branch.branchName and r1.branch.branchName != 'other branch')";
rootQueried = (Root) s.createQuery( qry ).uniqueResult();
assertEquals( root.rootName, rootQueried.rootName );
assertEquals( root.branch.branchName, rootQueried.branch.branchName );
assertEquals( leaf1.leafName, rootQueried.branch.leaves.get( 0 ).leafName );
assertEquals( leaf2.leafName, rootQueried.branch.leaves.get( 1 ).leafName );
s.getTransaction().commit();
s.close();
} }
} }