HHH-14439 Test executing the same query with subselects a second time with different list parameters
Signed-off-by: Yoann Rodière <yoann@hibernate.org>
This commit is contained in:
parent
81071a4594
commit
fb079d077c
|
@ -0,0 +1,215 @@
|
|||
/*
|
||||
* 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.test.annotations.query;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
import javax.persistence.Entity;
|
||||
import javax.persistence.FetchType;
|
||||
import javax.persistence.Id;
|
||||
import javax.persistence.ManyToOne;
|
||||
import javax.persistence.OneToMany;
|
||||
import javax.persistence.TypedQuery;
|
||||
|
||||
import org.hibernate.Hibernate;
|
||||
import org.hibernate.annotations.Fetch;
|
||||
import org.hibernate.annotations.FetchMode;
|
||||
import org.hibernate.cfg.Configuration;
|
||||
|
||||
import org.hibernate.testing.TestForIssue;
|
||||
import org.hibernate.testing.jdbc.SQLStatementInterceptor;
|
||||
import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
|
||||
import org.junit.Test;
|
||||
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
|
||||
/**
|
||||
* Tests the handling and expansion of list parameters,
|
||||
* particularly when using {@code @Fetch(FetchMode.SUBSELECT)}
|
||||
* (because this fetch mode involves building a map of parameters).
|
||||
*/
|
||||
public class QueryListParametersWithFetchSubSelectTest extends BaseCoreFunctionalTestCase {
|
||||
|
||||
private SQLStatementInterceptor sqlStatementInterceptor;
|
||||
|
||||
@Override
|
||||
protected void configure(Configuration configuration) {
|
||||
sqlStatementInterceptor = new SQLStatementInterceptor( configuration );
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Class<?>[] getAnnotatedClasses() {
|
||||
return new Class<?>[] { Parent.class, Child.class };
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void afterSessionFactoryBuilt() {
|
||||
inTransaction( s -> {
|
||||
for ( int i = 0; i < 10; i++ ) {
|
||||
Parent parent = new Parent( i );
|
||||
s.persist( parent );
|
||||
for ( int j = 0; j < 10; j++ ) {
|
||||
Child child = new Child( i * 100 + j, parent );
|
||||
parent.children.add( child );
|
||||
s.persist( child );
|
||||
}
|
||||
}
|
||||
} );
|
||||
}
|
||||
|
||||
@Test
|
||||
public void simple() {
|
||||
sqlStatementInterceptor.clear();
|
||||
|
||||
inTransaction( s -> {
|
||||
TypedQuery<Parent> query = s.createQuery( "select p from Parent p where id in :ids", Parent.class );
|
||||
query.setParameter( "ids", Arrays.asList( 0, 1, 2 ) );
|
||||
List<Parent> results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 0, 1, 2 );
|
||||
} );
|
||||
|
||||
// If we get here, children were initialized eagerly.
|
||||
// Did ORM actually use subselects?
|
||||
assertThat( sqlStatementInterceptor.getSqlQueries() ).hasSize( 2 );
|
||||
}
|
||||
|
||||
@Test
|
||||
@TestForIssue(jiraKey = "HHH-14439")
|
||||
public void reusingQueryWithFewerNamedParameters() {
|
||||
sqlStatementInterceptor.clear();
|
||||
|
||||
inTransaction( s -> {
|
||||
TypedQuery<Parent> query = s.createQuery( "select p from Parent p where id in :ids", Parent.class );
|
||||
|
||||
query.setParameter( "ids", Arrays.asList( 0, 1, 2, 3 ) );
|
||||
List<Parent> results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 0, 1, 2, 3 );
|
||||
|
||||
query.setParameter( "ids", Arrays.asList( 4, 5, 6 ) );
|
||||
results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 4, 5, 6 );
|
||||
|
||||
query.setParameter( "ids", Arrays.asList( 7, 8 ) );
|
||||
results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 7, 8 );
|
||||
} );
|
||||
|
||||
// If we get here, children were initialized eagerly.
|
||||
// Did ORM actually use subselects?
|
||||
assertThat( sqlStatementInterceptor.getSqlQueries() ).hasSize( 3 * 2 );
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
@TestForIssue(jiraKey = "HHH-14439")
|
||||
public void reusingQueryWithFewerOrdinalParameters() {
|
||||
sqlStatementInterceptor.clear();
|
||||
|
||||
inTransaction( s -> {
|
||||
TypedQuery<Parent> query = s.createQuery( "select p from Parent p where id in ?0", Parent.class );
|
||||
|
||||
query.setParameter( 0, Arrays.asList( 0, 1, 2, 3 ) );
|
||||
List<Parent> results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 0, 1, 2, 3 );
|
||||
|
||||
query.setParameter( 0, Arrays.asList( 4, 5, 6 ) );
|
||||
results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 4, 5, 6 );
|
||||
|
||||
query.setParameter( 0, Arrays.asList( 7, 8 ) );
|
||||
results = query.getResultList();
|
||||
assertThat( results )
|
||||
.allSatisfy( parent -> assertThat( Hibernate.isInitialized( parent.getChildren() ) ).isTrue() )
|
||||
.extracting( Parent::getId ).containsExactly( 7, 8 );
|
||||
} );
|
||||
|
||||
// If we get here, children were initialized eagerly.
|
||||
// Did ORM actually use subselects?
|
||||
assertThat( sqlStatementInterceptor.getSqlQueries() ).hasSize( 3 * 2 );
|
||||
}
|
||||
|
||||
@Entity(name = "Parent")
|
||||
public static class Parent {
|
||||
@Id
|
||||
private Integer id;
|
||||
|
||||
@OneToMany(mappedBy = "parent", fetch = FetchType.EAGER)
|
||||
@Fetch(FetchMode.SUBSELECT)
|
||||
private List<Child> children = new ArrayList<>();
|
||||
|
||||
public Parent() {
|
||||
}
|
||||
|
||||
public Parent(Integer id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public Integer getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(Integer id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public List<Child> getChildren() {
|
||||
return children;
|
||||
}
|
||||
|
||||
public void setChildren(List<Child> children) {
|
||||
this.children = children;
|
||||
}
|
||||
}
|
||||
|
||||
@Entity(name = "Child")
|
||||
public static class Child {
|
||||
|
||||
@Id
|
||||
private Integer id;
|
||||
|
||||
@ManyToOne
|
||||
private Parent parent;
|
||||
|
||||
public Child() {
|
||||
}
|
||||
|
||||
public Child(Integer id, Parent parent) {
|
||||
this.id = id;
|
||||
this.parent = parent;
|
||||
}
|
||||
|
||||
public Integer getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(Integer id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public Parent getParent() {
|
||||
return parent;
|
||||
}
|
||||
|
||||
public void setParent(Parent parent) {
|
||||
this.parent = parent;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue