Skip to content

Commit bfffcea

Browse files
committed
Use implicit join to create Predicate if possible
Fix GH-3349
1 parent 8571056 commit bfffcea

File tree

3 files changed

+108
-2
lines changed

3 files changed

+108
-2
lines changed

spring-data-jpa/src/main/java/org/springframework/data/jpa/repository/query/JpaQueryCreator.java

+1-2
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,6 @@
5656
* @author Moritz Becker
5757
* @author Andrey Kovalev
5858
* @author Greg Turnquist
59-
* @author Jinmyeong Kim
6059
*/
6160
public class JpaQueryCreator extends AbstractQueryCreator<CriteriaQuery<? extends Object>, Predicate> {
6261

@@ -390,7 +389,7 @@ private Expression<? extends Comparable> getComparablePath(Root<?> root, Part pa
390389
}
391390

392391
private <T> Expression<T> getTypedPath(Root<?> root, Part part) {
393-
return toExpressionRecursively(root, part.getProperty());
392+
return toExpressionRecursivelyForPredicate(root, part.getProperty());
394393
}
395394

396395
private <T> Expression<T> traversePath(Path<?> root, PropertyPath path) {

spring-data-jpa/src/main/java/org/springframework/data/jpa/repository/query/QueryUtils.java

+26
Original file line numberDiff line numberDiff line change
@@ -732,6 +732,32 @@ private static jakarta.persistence.criteria.Order toJpaOrder(Order order, From<?
732732
}
733733
}
734734

735+
/**
736+
* Creates an expression with joins by recursively navigating the path for constructing {@code Predicate},
737+
* it will use implicit join if possible to eliminate unnecessary join.
738+
*
739+
* @param from the {@link From}
740+
* @param property the property path
741+
* @param <T> the type of the expression
742+
* @return the expression
743+
*/
744+
@SuppressWarnings("unchecked")
745+
static <T> Expression<T> toExpressionRecursivelyForPredicate(From<?, ?> from, PropertyPath property) {
746+
747+
// see https://docs.jboss.org/hibernate/orm/current/userguide/html_single/Hibernate_User_Guide.html#hql-implicit-join
748+
Path<?> path = from;
749+
while (!property.isCollection()) {
750+
path = path.get(property.getSegment());
751+
if (property.hasNext()) {
752+
property = Objects.requireNonNull(property.next(), "An element of the property path is null");
753+
} else {
754+
return (Expression<T>) path;
755+
}
756+
}
757+
758+
return toExpressionRecursively(from, property);
759+
}
760+
735761
static <T> Expression<T> toExpressionRecursively(From<?, ?> from, PropertyPath property) {
736762
return toExpressionRecursively(from, property, false);
737763
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
/*
2+
* Copyright 2017-2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.jpa.repository.query;
17+
18+
import jakarta.persistence.EntityManager;
19+
import jakarta.persistence.PersistenceContext;
20+
import jakarta.persistence.TypedQuery;
21+
import org.hibernate.query.spi.SqmQuery;
22+
import org.hibernate.query.sqm.tree.from.SqmRoot;
23+
import org.hibernate.query.sqm.tree.select.SqmQuerySpec;
24+
import org.hibernate.query.sqm.tree.select.SqmSelectStatement;
25+
import org.junit.jupiter.api.Test;
26+
import org.junit.jupiter.api.extension.ExtendWith;
27+
import org.springframework.data.jpa.domain.sample.User;
28+
import org.springframework.data.jpa.provider.PersistenceProvider;
29+
import org.springframework.data.projection.SpelAwareProxyProjectionFactory;
30+
import org.springframework.data.repository.Repository;
31+
import org.springframework.data.repository.core.support.AbstractRepositoryMetadata;
32+
import org.springframework.data.repository.query.parser.PartTree;
33+
import org.springframework.test.context.ContextConfiguration;
34+
import org.springframework.test.context.junit.jupiter.SpringExtension;
35+
36+
import java.lang.reflect.Method;
37+
import java.util.List;
38+
39+
import static org.assertj.core.api.Assertions.assertThat;
40+
41+
/**
42+
* Integration tests for {@link JpaQueryCreator}.
43+
*
44+
* @author Yanming Zhou
45+
*/
46+
@ExtendWith(SpringExtension.class)
47+
@ContextConfiguration("classpath:infrastructure.xml")
48+
class JpaQueryCreatorIntegrationTests {
49+
50+
@PersistenceContext
51+
EntityManager entityManager;
52+
53+
@Test // GH-3349
54+
void implicitJoin() throws Exception {
55+
56+
Method method = SomeRepository.class.getMethod("findByManagerId", Integer.class);
57+
58+
PersistenceProvider provider = PersistenceProvider.fromEntityManager(entityManager);
59+
JpaQueryMethod queryMethod = new JpaQueryMethod(method,
60+
AbstractRepositoryMetadata.getMetadata(SomeRepository.class), new SpelAwareProxyProjectionFactory(), provider);
61+
62+
PartTree tree = new PartTree("findByManagerId", User.class);
63+
ParameterMetadataProvider metadataProvider = new ParameterMetadataProvider(entityManager.getCriteriaBuilder(),
64+
queryMethod.getParameters(), EscapeCharacter.DEFAULT);
65+
66+
JpaQueryCreator creator = new JpaQueryCreator(tree, queryMethod.getResultProcessor().getReturnedType(),
67+
entityManager.getCriteriaBuilder(), metadataProvider);
68+
69+
TypedQuery<?> query = entityManager.createQuery(creator.createQuery());
70+
SqmQuery sqmQuery = ((SqmQuery) query);
71+
SqmSelectStatement<?> statement = (SqmSelectStatement<?>) sqmQuery.getSqmStatement();
72+
SqmQuerySpec<?> spec = (SqmQuerySpec<?>) statement.getQueryPart();
73+
SqmRoot<?> root = spec.getFromClause().getRoots().get(0);
74+
75+
assertThat(root.getJoins()).isEmpty();
76+
}
77+
78+
interface SomeRepository extends Repository<User, Integer> {
79+
List<User> findByManagerId(Integer managerId);
80+
}
81+
}

0 commit comments

Comments
 (0)