spring-data-jpa QueryByExamplePredicateBuilder 源码
spring-data-jpa QueryByExamplePredicateBuilder 代码
文件路径:/spring-data-jpa/src/main/java/org/springframework/data/jpa/convert/QueryByExamplePredicateBuilder.java
/*
* Copyright 2016-2022 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.convert;
import jakarta.persistence.criteria.CriteriaBuilder;
import jakarta.persistence.criteria.Expression;
import jakarta.persistence.criteria.From;
import jakarta.persistence.criteria.Path;
import jakarta.persistence.criteria.Predicate;
import jakarta.persistence.criteria.Root;
import jakarta.persistence.metamodel.Attribute;
import jakarta.persistence.metamodel.Attribute.PersistentAttributeType;
import jakarta.persistence.metamodel.ManagedType;
import jakarta.persistence.metamodel.SingularAttribute;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import org.springframework.dao.InvalidDataAccessApiUsageException;
import org.springframework.data.domain.Example;
import org.springframework.data.domain.ExampleMatcher;
import org.springframework.data.domain.ExampleMatcher.PropertyValueTransformer;
import org.springframework.data.jpa.repository.query.EscapeCharacter;
import org.springframework.data.support.ExampleMatcherAccessor;
import org.springframework.data.util.DirectFieldAccessFallbackBeanWrapper;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.ObjectUtils;
import org.springframework.util.StringUtils;
/**
* {@link QueryByExamplePredicateBuilder} creates a single {@link CriteriaBuilder#and(Predicate...)} combined
* {@link Predicate} for a given {@link Example}. <br />
* The builder includes any {@link SingularAttribute} of the {@link Example#getProbe()} applying {@link String} and
* {@literal null} matching strategies configured on the {@link Example}. Ignored paths are no matter of their actual
* value not considered. <br />
*
* @author Christoph Strobl
* @author Mark Paluch
* @author Oliver Gierke
* @author Jens Schauder
* @author Greg Turnquist
* @since 1.10
*/
public class QueryByExamplePredicateBuilder {
private static final Set<PersistentAttributeType> ASSOCIATION_TYPES;
static {
ASSOCIATION_TYPES = EnumSet.of(PersistentAttributeType.MANY_TO_MANY, //
PersistentAttributeType.MANY_TO_ONE, //
PersistentAttributeType.ONE_TO_MANY, //
PersistentAttributeType.ONE_TO_ONE);
}
/**
* Extract the {@link Predicate} representing the {@link Example}.
*
* @param root must not be {@literal null}.
* @param cb must not be {@literal null}.
* @param example must not be {@literal null}.
* @return {@literal null} indicates no {@link Predicate}.
*/
@Nullable
public static <T> Predicate getPredicate(Root<T> root, CriteriaBuilder cb, Example<T> example) {
return getPredicate(root, cb, example, EscapeCharacter.DEFAULT);
}
/**
* Extract the {@link Predicate} representing the {@link Example}.
*
* @param root must not be {@literal null}.
* @param cb must not be {@literal null}.
* @param example must not be {@literal null}.
* @param escapeCharacter Must not be {@literal null}.
* @return {@literal null} indicates no constraints
*/
@Nullable
public static <T> Predicate getPredicate(Root<T> root, CriteriaBuilder cb, Example<T> example,
EscapeCharacter escapeCharacter) {
Assert.notNull(root, "Root must not be null");
Assert.notNull(cb, "CriteriaBuilder must not be null");
Assert.notNull(example, "Example must not be null");
ExampleMatcher matcher = example.getMatcher();
List<Predicate> predicates = getPredicates("", cb, root, root.getModel(), example.getProbe(),
example.getProbeType(), new ExampleMatcherAccessor(matcher), new PathNode("root", null, example.getProbe()),
escapeCharacter);
if (predicates.isEmpty()) {
return null;
}
if (predicates.size() == 1) {
return predicates.iterator().next();
}
Predicate[] array = predicates.toArray(new Predicate[0]);
return matcher.isAllMatching() ? cb.and(array) : cb.or(array);
}
@SuppressWarnings({ "rawtypes", "unchecked" })
static List<Predicate> getPredicates(String path, CriteriaBuilder cb, Path<?> from, ManagedType<?> type, Object value,
Class<?> probeType, ExampleMatcherAccessor exampleAccessor, PathNode currentNode,
EscapeCharacter escapeCharacter) {
List<Predicate> predicates = new ArrayList<>();
DirectFieldAccessFallbackBeanWrapper beanWrapper = new DirectFieldAccessFallbackBeanWrapper(value);
for (SingularAttribute attribute : type.getSingularAttributes()) {
String currentPath = !StringUtils.hasText(path) ? attribute.getName() : path + "." + attribute.getName();
if (exampleAccessor.isIgnoredPath(currentPath)) {
continue;
}
PropertyValueTransformer transformer = exampleAccessor.getValueTransformerForPath(currentPath);
Optional<Object> optionalValue = transformer
.apply(Optional.ofNullable(beanWrapper.getPropertyValue(attribute.getName())));
if (!optionalValue.isPresent()) {
if (exampleAccessor.getNullHandler().equals(ExampleMatcher.NullHandler.INCLUDE)) {
predicates.add(cb.isNull(from.get(attribute)));
}
continue;
}
Object attributeValue = optionalValue.get();
if (attributeValue == Optional.empty()) {
continue;
}
if (attribute.getPersistentAttributeType().equals(PersistentAttributeType.EMBEDDED)
|| (isAssociation(attribute) && !(from instanceof From))) {
predicates
.addAll(getPredicates(currentPath, cb, from.get(attribute.getName()), (ManagedType<?>) attribute.getType(),
attributeValue, probeType, exampleAccessor, currentNode, escapeCharacter));
continue;
}
if (isAssociation(attribute)) {
PathNode node = currentNode.add(attribute.getName(), attributeValue);
if (node.spansCycle()) {
throw new InvalidDataAccessApiUsageException(
String.format("Path '%s' from root %s must not span a cyclic property reference%n%s", currentPath,
ClassUtils.getShortName(probeType), node));
}
predicates.addAll(getPredicates(currentPath, cb, ((From<?, ?>) from).join(attribute.getName()),
(ManagedType<?>) attribute.getType(), attributeValue, probeType, exampleAccessor, node, escapeCharacter));
continue;
}
if (attribute.getJavaType().equals(String.class)) {
Expression<String> expression = from.get(attribute);
if (exampleAccessor.isIgnoreCaseForPath(currentPath)) {
expression = cb.lower(expression);
attributeValue = attributeValue.toString().toLowerCase();
}
switch (exampleAccessor.getStringMatcherForPath(currentPath)) {
case DEFAULT:
case EXACT:
predicates.add(cb.equal(expression, attributeValue));
break;
case CONTAINING:
predicates.add(cb.like( //
expression, //
"%" + escapeCharacter.escape(attributeValue.toString()) + "%", //
escapeCharacter.getEscapeCharacter() //
));
break;
case STARTING:
predicates.add(cb.like(//
expression, //
escapeCharacter.escape(attributeValue.toString()) + "%", //
escapeCharacter.getEscapeCharacter()) //
);
break;
case ENDING:
predicates.add(cb.like( //
expression, //
"%" + escapeCharacter.escape(attributeValue.toString()), //
escapeCharacter.getEscapeCharacter()) //
);
break;
default:
throw new IllegalArgumentException(
"Unsupported StringMatcher " + exampleAccessor.getStringMatcherForPath(currentPath));
}
} else {
predicates.add(cb.equal(from.get(attribute), attributeValue));
}
}
return predicates;
}
private static boolean isAssociation(Attribute<?, ?> attribute) {
return ASSOCIATION_TYPES.contains(attribute.getPersistentAttributeType());
}
/**
* {@link PathNode} is used to dynamically grow a directed graph structure that allows to detect cycles within its
* direct predecessor nodes by comparing parent node values using {@link System#identityHashCode(Object)}.
*
* @author Christoph Strobl
*/
private static class PathNode {
String name;
@Nullable PathNode parent;
List<PathNode> siblings = new ArrayList<>();
@Nullable Object value;
PathNode(String edge, @Nullable PathNode parent, @Nullable Object value) {
this.name = edge;
this.parent = parent;
this.value = value;
}
PathNode add(String attribute, @Nullable Object value) {
PathNode node = new PathNode(attribute, this, value);
siblings.add(node);
return node;
}
boolean spansCycle() {
if (value == null) {
return false;
}
String identityHex = ObjectUtils.getIdentityHexString(value);
PathNode current = parent;
while (current != null) {
if (current.value != null && ObjectUtils.getIdentityHexString(current.value).equals(identityHex)) {
return true;
}
current = current.parent;
}
return false;
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
if (parent != null) {
sb.append(parent);
sb.append(" -");
sb.append(name);
sb.append("-> ");
}
sb.append("[{ ");
sb.append(ObjectUtils.nullSafeToString(value));
sb.append(" }]");
return sb.toString();
}
}
}
相关信息
相关文章
0
赞
热门推荐
-
2、 - 优质文章
-
3、 gate.io
-
8、 golang
-
9、 openharmony
-
10、 Vue中input框自动聚焦