Fix subquery throwing SqlTreeCreationException( Found un-correlated path usage in sub query)

This commit is contained in:
Andrea Boriero 2022-03-04 11:15:15 +01:00 committed by Andrea Boriero
parent 0a73425520
commit 6e6cc5f06e
2 changed files with 233 additions and 1 deletions

View File

@ -2851,7 +2851,8 @@ public abstract class BaseSqmToSqlAstConverter<T extends Statement> extends Base
if ( parentTableGroup == null ) { if ( parentTableGroup == null ) {
final TableGroup parent = fromClauseIndex.findTableGroupOnParents( parentPath.getNavigablePath() ); final TableGroup parent = fromClauseIndex.findTableGroupOnParents( parentPath.getNavigablePath() );
if ( parent != null ) { if ( parent != null ) {
throw new SqlTreeCreationException( "Found un-correlated path usage in sub query - " + parentPath ); fromClauseIndex.register( (SqmPath<?>) parentPath, parent );
return parent;
} }
throw new SqlTreeCreationException( "Could not locate TableGroup - " + parentPath.getNavigablePath() ); throw new SqlTreeCreationException( "Could not locate TableGroup - " + parentPath.getNavigablePath() );
} }

View File

@ -0,0 +1,231 @@
package org.hibernate.orm.test.jpa.compliance;
import java.util.List;
import org.hibernate.dialect.MySQLDialect;
import org.hibernate.testing.orm.junit.EntityManagerFactoryScope;
import org.hibernate.testing.orm.junit.Jpa;
import org.hibernate.testing.orm.junit.SkipForDialect;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import jakarta.persistence.Entity;
import jakarta.persistence.Id;
import jakarta.persistence.criteria.CriteriaBuilder;
import jakarta.persistence.criteria.CriteriaDelete;
import jakarta.persistence.criteria.CriteriaQuery;
import jakarta.persistence.criteria.CriteriaUpdate;
import jakarta.persistence.criteria.Root;
import jakarta.persistence.criteria.Subquery;
import jakarta.persistence.metamodel.EntityType;
import static org.hamcrest.CoreMatchers.is;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.jupiter.api.Assertions.assertNull;
@Jpa(
annotatedClasses = { CriteriaSubqueryTest.TestEntity.class }
)
public class CriteriaSubqueryTest {
@BeforeEach
public void setUp(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager -> {
TestEntity testEntity = new TestEntity( 1, "1", 10 );
TestEntity testEntity2 = new TestEntity( 2, "2", 17 );
TestEntity testEntity3 = new TestEntity( 3, "3", 22 );
TestEntity testEntity4 = new TestEntity( 4, "4", 38 );
entityManager.persist( testEntity );
entityManager.persist( testEntity2 );
entityManager.persist( testEntity3 );
entityManager.persist( testEntity4 );
}
);
}
@AfterEach
public void tearDown(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager ->
entityManager.createQuery( "delete from TestEntity" ).executeUpdate()
);
}
@Test
public void existsInSubqueryTest(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager -> {
final Integer expectedId = 2;
final CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
final CriteriaQuery<TestEntity> criteriaQuery = criteriaBuilder.createQuery( TestEntity.class );
final Root<TestEntity> from = criteriaQuery.from( TestEntity.class );
final EntityType<TestEntity> testEntityType = from.getModel();
final Subquery<TestEntity> subquery = criteriaQuery.subquery( TestEntity.class );
final Root<TestEntity> subqueryFrom = subquery.from( TestEntity.class );
assertThat( subqueryFrom.getModel().getName(), is( TestEntity.class.getSimpleName() ) );
subquery.where( criteriaBuilder.equal(
from.get( testEntityType.getSingularAttribute( "id", Integer.class ) ),
expectedId
) ).select( subqueryFrom );
criteriaQuery.where( criteriaBuilder.exists( subquery ) );
criteriaQuery.select( from );
final List<TestEntity> testEntities = entityManager.createQuery( criteriaQuery ).getResultList();
assertThat( testEntities.size(), is( 1 ) );
assertThat( testEntities.get( 0 ).getId(), is( expectedId ) );
}
);
}
@Test
public void subqueryCiteriaSelectTest(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager -> {
final Integer expectedId = 2;
final CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
final CriteriaQuery<TestEntity> criteriaQuery = criteriaBuilder.createQuery( TestEntity.class );
final Root<TestEntity> from = criteriaQuery.from( TestEntity.class );
final EntityType<TestEntity> testEntityType = from.getModel();
final Subquery<TestEntity> subquery = criteriaQuery.subquery( TestEntity.class );
final Root<TestEntity> subqueryFrom = subquery.from( testEntityType );
assertThat( subqueryFrom.getModel().getName(), is( testEntityType.getName() ) );
subquery.where( criteriaBuilder.equal(
from.get( testEntityType.getSingularAttribute( "id", Integer.class ) ),
expectedId
) ).select( subqueryFrom );
criteriaQuery.where( criteriaBuilder.exists( subquery ) );
criteriaQuery.select( from );
final List<TestEntity> testEntities = entityManager.createQuery( criteriaQuery ).getResultList();
assertThat( testEntities.size(), is( 1 ) );
assertThat( testEntities.get( 0 ).getId(), is( expectedId ) );
}
);
}
@Test
@SkipForDialect( dialectClass = MySQLDialect.class, matchSubTypes = true, reason = "does not support specifying in the subquery from clause the the same table used in the delete/update ")
public void subqueryCriteriaDeleteTest(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager -> {
final CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
final CriteriaDelete<TestEntity> criteriaDelete = criteriaBuilder.createCriteriaDelete( TestEntity.class );
final Root<TestEntity> from = criteriaDelete.from( TestEntity.class );
final EntityType<TestEntity> testEntityType = from.getModel();
final Subquery<TestEntity> subquery = criteriaDelete.subquery( TestEntity.class );
final Root<TestEntity> subqueryFrom = subquery.from( TestEntity.class );
subquery.where( criteriaBuilder.equal(
from.get( testEntityType.getSingularAttribute( "id", Integer.class ) ), 2 ) )
.select( subqueryFrom );
criteriaDelete.where( criteriaBuilder.exists( subquery ) );
final int entityDeleted = entityManager.createQuery( criteriaDelete ).executeUpdate();
assertThat( entityDeleted, is( 1 ) );
}
);
scope.inTransaction(
entityManager -> {
final TestEntity testEntity = entityManager.find( TestEntity.class, 2 );
assertNull( testEntity );
}
);
}
@Test
@SkipForDialect( dialectClass = MySQLDialect.class, matchSubTypes = true, reason = "does not support specifying in the subquery from clause the the same table used in the delete/update ")
public void subqueryCriteriaUpdateTest(EntityManagerFactoryScope scope) {
scope.inTransaction(
entityManager -> {
final CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
final CriteriaUpdate<TestEntity> criteriaUpdate = criteriaBuilder.createCriteriaUpdate( TestEntity.class );
final Root<TestEntity> from = criteriaUpdate.from( TestEntity.class );
final EntityType<TestEntity> testEntityType = from.getModel();
criteriaUpdate.set(
from.<Integer>get( "age" ),
criteriaBuilder.sum(
from.get( testEntityType.getSingularAttribute( "age", Integer.class ) ),
13
)
);
final Subquery<TestEntity> subquery = criteriaUpdate.subquery( TestEntity.class );
final Root<TestEntity> subqueryFrom = subquery.from( TestEntity.class );
subquery.where( criteriaBuilder.equal(
from.get( testEntityType.getSingularAttribute( "id", Integer.class ) ), 2 ) )
.select( subqueryFrom );
criteriaUpdate.where( criteriaBuilder.exists( subquery ) );
int entityUpdated = entityManager.createQuery( criteriaUpdate ).executeUpdate();
assertThat( entityUpdated, is( 1 ) );
}
);
scope.inTransaction(
entityManager -> {
TestEntity testEntity = entityManager.find( TestEntity.class, 2 );
assertThat( testEntity.getAge(), is( 30 ) );
}
);
}
@Entity(name = "TestEntity")
public static class TestEntity {
@Id
private Integer id;
private String name;
private Integer age;
public TestEntity() {
}
public TestEntity(Integer id, String name, Integer age) {
this.id = id;
this.name = name;
this.age = age;
}
public Integer getId() {
return id;
}
public Integer getAge() {
return age;
}
}
}