QuerydslJpaPredicateExecutor.java
/*
* Copyright 2008-2025 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.data.jpa.repository.support;
import jakarta.persistence.EntityManager;
import jakarta.persistence.LockModeType;
import java.util.List;
import java.util.Optional;
import java.util.function.BiFunction;
import java.util.function.Function;
import org.jspecify.annotations.Nullable;
import org.springframework.dao.IncorrectResultSizeDataAccessException;
import org.springframework.dao.InvalidDataAccessApiUsageException;
import org.springframework.data.domain.KeysetScrollPosition;
import org.springframework.data.domain.OffsetScrollPosition;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.data.domain.Sort.Order;
import org.springframework.data.jpa.repository.EntityGraph;
import org.springframework.data.jpa.repository.query.KeysetScrollDelegate;
import org.springframework.data.jpa.repository.query.KeysetScrollDelegate.QueryStrategy;
import org.springframework.data.jpa.repository.query.KeysetScrollSpecification;
import org.springframework.data.jpa.repository.support.FluentQuerySupport.ScrollQueryFactory;
import org.springframework.data.projection.ProjectionFactory;
import org.springframework.data.projection.SpelAwareProxyProjectionFactory;
import org.springframework.data.querydsl.EntityPathResolver;
import org.springframework.data.querydsl.QSort;
import org.springframework.data.querydsl.QuerydslPredicateExecutor;
import org.springframework.data.repository.query.FluentQuery;
import org.springframework.data.repository.query.FluentQuery.FetchableFluentQuery;
import org.springframework.data.support.PageableExecutionUtils;
import org.springframework.util.Assert;
import com.querydsl.core.NonUniqueResultException;
import com.querydsl.core.types.ConstantImpl;
import com.querydsl.core.types.EntityPath;
import com.querydsl.core.types.Expression;
import com.querydsl.core.types.NullExpression;
import com.querydsl.core.types.Ops;
import com.querydsl.core.types.OrderSpecifier;
import com.querydsl.core.types.Predicate;
import com.querydsl.core.types.dsl.BooleanExpression;
import com.querydsl.core.types.dsl.Expressions;
import com.querydsl.core.types.dsl.PathBuilder;
import com.querydsl.jpa.JPQLQuery;
import com.querydsl.jpa.impl.AbstractJPAQuery;
/**
* Querydsl specific fragment for extending {@link SimpleJpaRepository} with an implementation of
* {@link QuerydslPredicateExecutor}.
*
* @author Oliver Gierke
* @author Thomas Darimont
* @author Mark Paluch
* @author Jocelyn Ntakpe
* @author Christoph Strobl
* @author Jens Schauder
* @author Greg Turnquist
* @author Yanming Zhou
*/
public class QuerydslJpaPredicateExecutor<T> implements QuerydslPredicateExecutor<T>, JpaRepositoryConfigurationAware {
private static final String PREDICATE_MUST_NOT_BE_NULL = "Predicate must not be null";
private static final String ORDER_SPECIFIERS_MUST_NOT_BE_NULL = "Order specifiers must not be null";
private static final String SORT_MUST_NOT_BE_NULL = "Sort must not be null";
private static final String PAGEABLE_MUST_NOT_BE_NULL = "Pageable must not be null";
private static final String QUERY_FUNCTION_MUST_NOT_BE_NULL = "Query function must not be null";
private final JpaEntityInformation<T, ?> entityInformation;
private final EntityPath<T> path;
private final Querydsl querydsl;
private final QuerydslQueryStrategy scrollQueryAdapter;
private final EntityManager entityManager;
private @Nullable CrudMethodMetadata metadata;
private @Nullable ProjectionFactory projectionFactory;
/**
* Creates a new {@link QuerydslJpaPredicateExecutor} from the given domain class and {@link EntityManager} and uses
* the given {@link EntityPathResolver} to translate the domain class into an {@link EntityPath}.
*
* @param entityInformation must not be {@literal null}.
* @param entityManager must not be {@literal null}.
* @param resolver must not be {@literal null}.
* @param metadata maybe {@literal null}.
*/
public QuerydslJpaPredicateExecutor(JpaEntityInformation<T, ?> entityInformation, EntityManager entityManager,
EntityPathResolver resolver, @Nullable CrudMethodMetadata metadata) {
this.entityInformation = entityInformation;
this.metadata = metadata;
this.path = resolver.createPath(entityInformation.getJavaType());
this.querydsl = new Querydsl(entityManager, new PathBuilder<T>(path.getType(), path.getMetadata()));
this.entityManager = entityManager;
this.scrollQueryAdapter = new QuerydslQueryStrategy();
}
@Override
public void setRepositoryMethodMetadata(CrudMethodMetadata metadata) {
this.metadata = metadata;
}
@Override
public void setProjectionFactory(ProjectionFactory projectionFactory) {
this.projectionFactory = projectionFactory;
}
@Override
public Optional<T> findOne(Predicate predicate) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
try {
return Optional.ofNullable(createQuery(predicate).select(path).limit(2).fetchOne());
} catch (NonUniqueResultException ex) {
throw new IncorrectResultSizeDataAccessException(ex.getMessage(), 1, ex);
}
}
@Override
public List<T> findAll(Predicate predicate) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
return createQuery(predicate).select(path).fetch();
}
@Override
public List<T> findAll(Predicate predicate, OrderSpecifier<?>... orders) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
Assert.notNull(orders, ORDER_SPECIFIERS_MUST_NOT_BE_NULL);
return executeSorted(createQuery(predicate).select(path), orders);
}
@Override
public List<T> findAll(Predicate predicate, Sort sort) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
Assert.notNull(sort, SORT_MUST_NOT_BE_NULL);
return executeSorted(createQuery(predicate).select(path), sort);
}
@Override
public List<T> findAll(OrderSpecifier<?>... orders) {
Assert.notNull(orders, ORDER_SPECIFIERS_MUST_NOT_BE_NULL);
return executeSorted(createQuery(new Predicate[0]).select(path), orders);
}
@Override
public Page<T> findAll(Predicate predicate, Pageable pageable) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
Assert.notNull(pageable, PAGEABLE_MUST_NOT_BE_NULL);
final JPQLQuery<?> countQuery = createCountQuery(predicate);
JPQLQuery<T> query = querydsl.applyPagination(pageable, createQuery(predicate).select(path));
return PageableExecutionUtils.getPage(query.fetch(), pageable, countQuery::fetchCount);
}
@SuppressWarnings("unchecked")
@Override
public <S extends T, R extends @Nullable Object> R findBy(Predicate predicate,
Function<FetchableFluentQuery<S>, R> queryFunction) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
Assert.notNull(queryFunction, QUERY_FUNCTION_MUST_NOT_BE_NULL);
Function<Sort, AbstractJPAQuery<?, ?>> finder = sort -> {
AbstractJPAQuery<?, ?> select = (AbstractJPAQuery<?, ?>) createQuery(predicate).select(path);
if (sort != null) {
select = (AbstractJPAQuery<?, ?>) querydsl.applySorting(sort, select);
}
return select;
};
ScrollQueryFactory<AbstractJPAQuery<?, ?>> scroll = (q, scrollPosition) -> {
Predicate predicateToUse = predicate;
Sort sort = q.sort;
if (scrollPosition instanceof KeysetScrollPosition keyset) {
KeysetScrollDelegate delegate = KeysetScrollDelegate.of(keyset.getDirection());
sort = KeysetScrollSpecification.createSort(keyset, sort, entityInformation);
BooleanExpression keysetPredicate = delegate.createPredicate(keyset, sort, scrollQueryAdapter);
if (keysetPredicate != null) {
predicateToUse = predicate instanceof BooleanExpression be ? be.and(keysetPredicate)
: keysetPredicate.and(predicate);
}
}
AbstractJPAQuery<?, ?> select = (AbstractJPAQuery<?, ?>) createQuery(predicateToUse).select(path);
select = (AbstractJPAQuery<?, ?>) querydsl.applySorting(sort, select);
if (scrollPosition instanceof OffsetScrollPosition offset) {
if (!offset.isInitial()) {
select.offset(offset.getOffset() + 1);
}
}
return select;
};
BiFunction<Sort, Pageable, AbstractJPAQuery<?, ?>> pagedFinder = (sort, pageable) -> {
AbstractJPAQuery<?, ?> select = finder.apply(sort);
if (pageable.isPaged()) {
select = (AbstractJPAQuery<?, ?>) querydsl.applyPagination(pageable, select);
}
return select;
};
FetchableFluentQueryByPredicate<T, T> fluentQuery = new FetchableFluentQueryByPredicate<>( //
path, predicate, //
this.entityInformation, //
finder, //
scroll, //
pagedFinder, //
this::count, //
this::exists, //
entityManager, //
getProjectionFactory());
R result = queryFunction.apply((FetchableFluentQuery<S>) fluentQuery);
if (result instanceof FluentQuery<?>) {
throw new InvalidDataAccessApiUsageException(
"findBy(���) queries must result a query result and not the FluentQuery object to ensure that queries are executed within the scope of the findBy(���) method");
}
return result;
}
@Override
public long count(Predicate predicate) {
return createQuery(predicate).fetchCount();
}
/**
* Delete entities by the given {@link Predicate} by loading and removing these.
* <p>
* This method is useful for a small amount of entities. For large amounts of entities, consider using batch deletes
* by declaring a delete query yourself.
*
* @param predicate the {@link Predicate} to delete entities by, must not be {@literal null}.
* @return number of deleted entities.
* @since 4.0
*/
public long delete(Predicate predicate) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
List<T> results = (List<T>) createQuery(predicate).fetch();
int deleted = 0;
for (T entity : results) {
if (SimpleJpaRepository.doDelete(entityManager, entityInformation, entity)) {
deleted++;
}
}
return deleted;
}
@Override
public boolean exists(Predicate predicate) {
return createQuery(predicate).select(Expressions.ONE).fetchFirst() != null;
}
/**
* Creates a new {@link JPQLQuery} for the given {@link Predicate}.
*
* @param predicate
* @return the Querydsl {@link JPQLQuery}.
*/
protected AbstractJPAQuery<?, ?> createQuery(Predicate... predicate) {
Assert.notNull(predicate, PREDICATE_MUST_NOT_BE_NULL);
AbstractJPAQuery<?, ?> query = doCreateQuery(getQueryHints().withFetchGraphs(entityManager), predicate);
CrudMethodMetadata metadata = getRepositoryMethodMetadata();
if (metadata == null) {
return query;
}
LockModeType type = metadata.getLockModeType();
return type == null ? query : query.setLockMode(type);
}
/**
* Creates a new {@link JPQLQuery} count query for the given {@link Predicate}.
*
* @param predicate, can be {@literal null}.
* @return the Querydsl count {@link JPQLQuery}.
*/
protected JPQLQuery<?> createCountQuery(@Nullable Predicate... predicate) {
return doCreateQuery(getQueryHintsForCount(), predicate);
}
private @Nullable CrudMethodMetadata getRepositoryMethodMetadata() {
return metadata;
}
/**
* Returns {@link QueryHints} with the query hints based on the current {@link CrudMethodMetadata} and potential
* {@link EntityGraph} information.
*
* @return
*/
private QueryHints getQueryHints() {
return metadata == null ? QueryHints.NoHints.INSTANCE : DefaultQueryHints.of(entityInformation, metadata);
}
/**
* Returns {@link QueryHints} with the query hints based on the current {@link CrudMethodMetadata} and potential
* {@link EntityGraph} information and filtered for those hints that are to be applied to count queries.
*
* @return
*/
private QueryHints getQueryHintsForCount() {
return metadata == null ? QueryHints.NoHints.INSTANCE
: DefaultQueryHints.of(entityInformation, metadata).forCounts();
}
private AbstractJPAQuery<?, ?> doCreateQuery(QueryHints hints, @Nullable Predicate... predicate) {
AbstractJPAQuery<?, ?> query = querydsl.createQuery(path);
if (predicate != null) {
query = query.where(predicate);
}
hints.forEach(query::setHint);
return query;
}
/**
* Executes the given {@link JPQLQuery} after applying the given {@link OrderSpecifier}s.
*
* @param query must not be {@literal null}.
* @param orders must not be {@literal null}.
* @return
*/
private List<T> executeSorted(JPQLQuery<T> query, OrderSpecifier<?>... orders) {
return executeSorted(query, new QSort(orders));
}
/**
* Executes the given {@link JPQLQuery} after applying the given {@link Sort}.
*
* @param query must not be {@literal null}.
* @param sort must not be {@literal null}.
* @return
*/
private List<T> executeSorted(JPQLQuery<T> query, Sort sort) {
return querydsl.applySorting(sort, query).fetch();
}
private ProjectionFactory getProjectionFactory() {
if (projectionFactory == null) {
projectionFactory = new SpelAwareProxyProjectionFactory();
}
return projectionFactory;
}
class QuerydslQueryStrategy implements QueryStrategy<Expression<?>, BooleanExpression> {
@Override
public Expression<?> createExpression(String property) {
return querydsl.createExpression(property);
}
@Override
public BooleanExpression compare(Order order, Expression<?> propertyExpression, @Nullable Object value) {
if (value == null) {
return Expressions.booleanOperation(order.isAscending() ? Ops.IS_NULL : Ops.IS_NOT_NULL, propertyExpression);
}
return Expressions.booleanOperation(order.isAscending() ? Ops.GT : Ops.LT, propertyExpression,
ConstantImpl.create(value));
}
@Override
public BooleanExpression compare(String property, Expression<?> propertyExpression, @Nullable Object value) {
return Expressions.booleanOperation(Ops.EQ, propertyExpression,
value == null ? NullExpression.DEFAULT : ConstantImpl.create(value));
}
@Override
public BooleanExpression and(List<BooleanExpression> intermediate) {
return Expressions.allOf(intermediate.toArray(new BooleanExpression[0]));
}
@Override
public BooleanExpression or(List<BooleanExpression> intermediate) {
return Expressions.anyOf(intermediate.toArray(new BooleanExpression[0]));
}
}
}