HHH-15648 Fix fetching association and using implicit path in condition
This commit is contained in:
parent
60e81a7a4d
commit
a471bbea8c
|
@ -193,10 +193,17 @@ public class EntityValuedPathInterpretation<T> extends AbstractSqmPathInterpreta
|
|||
.findTableGroup( tableGroup.getNavigablePath().getParent() );
|
||||
}
|
||||
else {
|
||||
if ( isCorrelated( tableGroup, sqlAstCreationState ) ) {
|
||||
if ( isCorrelated( tableGroup, sqlAstCreationState )
|
||||
|| !tableGroup.getNavigablePath().isParentOrEqual( navigablePath ) ) {
|
||||
// Access to the parent table group is forbidden for correlated table groups. For more details,
|
||||
// see: `ToOneAttributeMapping.createRootTableGroupJoin`
|
||||
// Due to that, we forcefully use the model part to which this association points to i.e. the target
|
||||
|
||||
// Also force the use of the FK target key if the navigable path for this entity valued path is
|
||||
// not equal to or a child of the table group navigable path.
|
||||
// This can happen when using an implicit join path e.g. `where root.association.id is null`,
|
||||
// yet also an explicit join was made which is compatible e.g. `join fetch root.association`.
|
||||
// Since we have an explicit join in this case anyway, it's fine to us the FK target key.
|
||||
resultModelPart = associationMapping.getForeignKeyDescriptor()
|
||||
.getPart( associationMapping.getSideNature().inverse() );
|
||||
}
|
||||
|
|
|
@ -192,6 +192,19 @@ public class NavigablePath implements DotIdentifierSequence, Serializable {
|
|||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determine whether this path is part of the given path's parent
|
||||
*/
|
||||
public boolean isParentOrEqual(NavigablePath navigablePath) {
|
||||
while ( navigablePath != null ) {
|
||||
if ( this.equals( navigablePath ) ) {
|
||||
return true;
|
||||
}
|
||||
navigablePath = navigablePath.getParent();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public boolean pathsMatch(NavigablePath p) {
|
||||
return this == p || p != null && localName.equals( p.localName )
|
||||
&& ( parent == null ? p.parent == null && Objects.equals( alias, p.alias ) : parent.pathsMatch( p.parent ) );
|
||||
|
|
|
@ -0,0 +1,109 @@
|
|||
/*
|
||||
* Hibernate, Relational Persistence for Idiomatic Java
|
||||
*
|
||||
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||
*/
|
||||
|
||||
package org.hibernate.orm.test.hql;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
import org.hibernate.query.criteria.HibernateCriteriaBuilder;
|
||||
import org.hibernate.query.criteria.JpaCriteriaQuery;
|
||||
import org.hibernate.query.criteria.JpaRoot;
|
||||
|
||||
import org.hibernate.testing.TestForIssue;
|
||||
import org.hibernate.testing.jdbc.SQLStatementInspector;
|
||||
import org.hibernate.testing.orm.junit.DomainModel;
|
||||
import org.hibernate.testing.orm.junit.SessionFactory;
|
||||
import org.hibernate.testing.orm.junit.SessionFactoryScope;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import jakarta.persistence.Entity;
|
||||
import jakarta.persistence.Id;
|
||||
import jakarta.persistence.ManyToOne;
|
||||
import jakarta.persistence.Table;
|
||||
import jakarta.persistence.criteria.Join;
|
||||
import jakarta.persistence.criteria.JoinType;
|
||||
|
||||
/**
|
||||
* @author Christian Beikov
|
||||
*/
|
||||
@DomainModel(
|
||||
annotatedClasses = {
|
||||
ManyToOneJoinReuseTest.Book.class,
|
||||
ManyToOneJoinReuseTest.BookList.class
|
||||
}
|
||||
)
|
||||
@SessionFactory(statementInspectorClass = SQLStatementInspector.class)
|
||||
public class ManyToOneJoinReuseTest {
|
||||
|
||||
@Test
|
||||
@TestForIssue(jiraKey = "HHH-15648")
|
||||
public void fetchAndImplicitPath(SessionFactoryScope scope) {
|
||||
SQLStatementInspector sqlStatementInterceptor = (SQLStatementInspector) scope.getStatementInspector();
|
||||
sqlStatementInterceptor.clear();
|
||||
scope.inTransaction(
|
||||
session -> {
|
||||
HibernateCriteriaBuilder cb = session.getCriteriaBuilder();
|
||||
JpaCriteriaQuery<BookList> query = cb.createQuery( BookList.class );
|
||||
|
||||
JpaRoot<BookList> root = query.from( BookList.class );
|
||||
root.fetch( "book", JoinType.INNER );
|
||||
query.where( root.get( "book" ).isNotNull() );
|
||||
|
||||
session.createQuery( query ).getResultList();
|
||||
sqlStatementInterceptor.assertExecuted( "select b1_0.id,b2_0.isbn,b2_0.title from BookList b1_0 join book b2_0 on b2_0.isbn=b1_0.book_isbn where b2_0.isbn is not null" );
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
@Test
|
||||
@TestForIssue(jiraKey = "HHH-15645")
|
||||
public void joinAndImplicitPath(SessionFactoryScope scope) {
|
||||
SQLStatementInspector sqlStatementInterceptor = (SQLStatementInspector) scope.getStatementInspector();
|
||||
sqlStatementInterceptor.clear();
|
||||
scope.inTransaction(
|
||||
session -> {
|
||||
HibernateCriteriaBuilder cb = session.getCriteriaBuilder();
|
||||
JpaCriteriaQuery<BookList> query = cb.createQuery( BookList.class );
|
||||
|
||||
JpaRoot<BookList> root = query.from( BookList.class );
|
||||
Join<Object, Object> join = root.join( "book", JoinType.INNER );
|
||||
query.where(
|
||||
cb.and(
|
||||
root.get( "book" ).isNotNull(),
|
||||
join.isNotNull()
|
||||
)
|
||||
);
|
||||
|
||||
session.createQuery( query ).getResultList();
|
||||
sqlStatementInterceptor.assertExecuted( "select b1_0.id,b1_0.book_isbn from BookList b1_0 join book b2_0 on b2_0.isbn=b1_0.book_isbn where b2_0.isbn is not null and b1_0.book_isbn is not null" );
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
@Entity(name = "BookList")
|
||||
@Table
|
||||
public static class BookList {
|
||||
|
||||
@Id
|
||||
private String id;
|
||||
|
||||
@ManyToOne
|
||||
private Book book;
|
||||
|
||||
}
|
||||
|
||||
@Entity(name = "Book")
|
||||
@Table(name = "book")
|
||||
public static class Book {
|
||||
|
||||
@Id
|
||||
private String isbn;
|
||||
|
||||
private String title;
|
||||
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue