HHH-9090 : HQL parser is trying to reuse parent implied join for subquery
This commit is contained in:
parent
b1c02c85cb
commit
153c4e32ef
|
@ -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 ) {
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue