提交 86c86151 编写于 作者: R Rossen Stoyanchev

Accept Predicate instead of HandlerTypePredicate

Issue: SPR-16336
上级 0092653d
...@@ -29,14 +29,19 @@ import org.springframework.util.ClassUtils; ...@@ -29,14 +29,19 @@ import org.springframework.util.ClassUtils;
import org.springframework.util.StringUtils; import org.springframework.util.StringUtils;
/** /**
* A {@code Predicate} to match request handling component types based on the * A {@code Predicate} to match request handling component types if
* following selectors: * <strong>any</strong> of the following selectors match:
* <ul> * <ul>
* <li>Base packages -- for selecting handlers by their package. * <li>Base packages -- for selecting handlers by their package.
* <li>Assignable types -- for selecting handlers by super type. * <li>Assignable types -- for selecting handlers by super type.
* <li>Annotations -- for selecting handlers annotated in a specific way. * <li>Annotations -- for selecting handlers annotated in a specific way.
* </ul> * </ul>
* <p>Use static factory methods in this class to create a Predicate. * <p>Composability methods on {@link Predicate} can be used :
* <pre class="code">
* Predicate&lt;Class&lt;?&gt;&gt; predicate =
* HandlerTypePredicate.forAnnotation(RestController)
* .and(HandlerTypePredicate.forBasePackage("org.example"));
* </pre>
* *
* @author Rossen Stoyanchev * @author Rossen Stoyanchev
* @since 5.1 * @since 5.1
......
/*
* Copyright 2002-2018 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
*
* http://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.web.method;
import java.util.function.Predicate;
import org.junit.Test;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RestController;
import static org.junit.Assert.*;
/**
* Unit tests for {@link HandlerTypePredicate}.
* @author Rossen Stoyanchev
*/
public class HandlerTypePredicateTests {
@Test
public void forAnnotation() {
Predicate<Class<?>> predicate = HandlerTypePredicate.forAnnotation(Controller.class);
assertTrue(predicate.test(HtmlController.class));
assertTrue(predicate.test(ApiController.class));
assertTrue(predicate.test(AnotherApiController.class));
}
@Test
public void forAnnotationWithException() {
Predicate<Class<?>> predicate = HandlerTypePredicate.forAnnotation(Controller.class)
.and(HandlerTypePredicate.forAssignableType(Special.class));
assertFalse(predicate.test(HtmlController.class));
assertFalse(predicate.test(ApiController.class));
assertTrue(predicate.test(AnotherApiController.class));
}
@Controller
private static class HtmlController {}
@RestController
private static class ApiController {}
@RestController
private static class AnotherApiController implements Special {}
interface Special {}
}
...@@ -18,9 +18,9 @@ package org.springframework.web.reactive.config; ...@@ -18,9 +18,9 @@ package org.springframework.web.reactive.config;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.function.Predicate;
import org.springframework.lang.Nullable; import org.springframework.lang.Nullable;
import org.springframework.web.method.HandlerTypePredicate;
/** /**
* Assist with configuring {@code HandlerMapping}'s with path matching options. * Assist with configuring {@code HandlerMapping}'s with path matching options.
...@@ -39,7 +39,7 @@ public class PathMatchConfigurer { ...@@ -39,7 +39,7 @@ public class PathMatchConfigurer {
private Boolean caseSensitiveMatch; private Boolean caseSensitiveMatch;
@Nullable @Nullable
private Map<String, HandlerTypePredicate> pathPrefixes; private Map<String, Predicate<Class<?>>> pathPrefixes;
/** /**
...@@ -66,13 +66,14 @@ public class PathMatchConfigurer { ...@@ -66,13 +66,14 @@ public class PathMatchConfigurer {
* Configure a path prefix to apply to matching controller methods. * Configure a path prefix to apply to matching controller methods.
* <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping} * <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping}
* method whose controller type is matched by the corresponding * method whose controller type is matched by the corresponding
* {@link HandlerTypePredicate}. The prefix for the first matching predicate * {@code Predicate}. The prefix for the first matching predicate is used.
* is used. * <p>Consider using {@link org.springframework.web.method.HandlerTypePredicate
* HandlerTypePredicate} to group controllers.
* @param prefix the path prefix to apply * @param prefix the path prefix to apply
* @param predicate a predicate for matching controller types * @param predicate a predicate for matching controller types
* @since 5.1 * @since 5.1
*/ */
public PathMatchConfigurer addPathPrefix(String prefix, HandlerTypePredicate predicate) { public PathMatchConfigurer addPathPrefix(String prefix, Predicate<Class<?>> predicate) {
this.pathPrefixes = this.pathPrefixes == null ? new LinkedHashMap<>() : this.pathPrefixes; this.pathPrefixes = this.pathPrefixes == null ? new LinkedHashMap<>() : this.pathPrefixes;
this.pathPrefixes.put(prefix, predicate); this.pathPrefixes.put(prefix, predicate);
return this; return this;
...@@ -89,7 +90,7 @@ public class PathMatchConfigurer { ...@@ -89,7 +90,7 @@ public class PathMatchConfigurer {
} }
@Nullable @Nullable
protected Map<String, HandlerTypePredicate> getPathPrefixes() { protected Map<String, Predicate<Class<?>>> getPathPrefixes() {
return this.pathPrefixes; return this.pathPrefixes;
} }
} }
...@@ -18,6 +18,7 @@ package org.springframework.web.reactive.config; ...@@ -18,6 +18,7 @@ package org.springframework.web.reactive.config;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.function.Predicate;
import reactor.core.publisher.Mono; import reactor.core.publisher.Mono;
...@@ -44,7 +45,6 @@ import org.springframework.validation.Validator; ...@@ -44,7 +45,6 @@ import org.springframework.validation.Validator;
import org.springframework.web.bind.WebDataBinder; import org.springframework.web.bind.WebDataBinder;
import org.springframework.web.bind.support.ConfigurableWebBindingInitializer; import org.springframework.web.bind.support.ConfigurableWebBindingInitializer;
import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.method.HandlerTypePredicate;
import org.springframework.web.reactive.DispatcherHandler; import org.springframework.web.reactive.DispatcherHandler;
import org.springframework.web.reactive.HandlerMapping; import org.springframework.web.reactive.HandlerMapping;
import org.springframework.web.reactive.accept.RequestedContentTypeResolver; import org.springframework.web.reactive.accept.RequestedContentTypeResolver;
...@@ -131,7 +131,7 @@ public class WebFluxConfigurationSupport implements ApplicationContextAware { ...@@ -131,7 +131,7 @@ public class WebFluxConfigurationSupport implements ApplicationContextAware {
mapping.setUseCaseSensitiveMatch(useCaseSensitiveMatch); mapping.setUseCaseSensitiveMatch(useCaseSensitiveMatch);
} }
Map<String, HandlerTypePredicate> pathPrefixes = configurer.getPathPrefixes(); Map<String, Predicate<Class<?>>> pathPrefixes = configurer.getPathPrefixes();
if (pathPrefixes != null) { if (pathPrefixes != null) {
mapping.setPathPrefixes(pathPrefixes); mapping.setPathPrefixes(pathPrefixes);
} }
......
...@@ -21,6 +21,7 @@ import java.lang.reflect.Method; ...@@ -21,6 +21,7 @@ import java.lang.reflect.Method;
import java.util.Collections; import java.util.Collections;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.function.Predicate;
import org.springframework.context.EmbeddedValueResolverAware; import org.springframework.context.EmbeddedValueResolverAware;
import org.springframework.core.annotation.AnnotatedElementUtils; import org.springframework.core.annotation.AnnotatedElementUtils;
...@@ -34,7 +35,6 @@ import org.springframework.web.bind.annotation.CrossOrigin; ...@@ -34,7 +35,6 @@ import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.method.HandlerTypePredicate;
import org.springframework.web.method.HandlerMethod; import org.springframework.web.method.HandlerMethod;
import org.springframework.web.reactive.accept.RequestedContentTypeResolver; import org.springframework.web.reactive.accept.RequestedContentTypeResolver;
import org.springframework.web.reactive.accept.RequestedContentTypeResolverBuilder; import org.springframework.web.reactive.accept.RequestedContentTypeResolverBuilder;
...@@ -53,7 +53,7 @@ import org.springframework.web.reactive.result.method.RequestMappingInfoHandlerM ...@@ -53,7 +53,7 @@ import org.springframework.web.reactive.result.method.RequestMappingInfoHandlerM
public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMapping public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMapping
implements EmbeddedValueResolverAware { implements EmbeddedValueResolverAware {
private final Map<String, HandlerTypePredicate> pathPrefixes = new LinkedHashMap<>(); private final Map<String, Predicate<Class<?>>> pathPrefixes = new LinkedHashMap<>();
private RequestedContentTypeResolver contentTypeResolver = new RequestedContentTypeResolverBuilder().build(); private RequestedContentTypeResolver contentTypeResolver = new RequestedContentTypeResolverBuilder().build();
...@@ -66,13 +66,16 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -66,13 +66,16 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
/** /**
* Configure path prefixes to apply to controller methods. * Configure path prefixes to apply to controller methods.
* <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping} * <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping}
* method whose controller type is matched by the corresponding * method whose controller type is matched by a corresponding
* {@link HandlerTypePredicate} in the map. The prefix for the first matching * {@code Predicate} in the map. The prefix for the first matching predicate
* predicate is used, assuming the input map has predictable order. * is used, assuming the input map has predictable order.
* <p>Consider using {@link org.springframework.web.method.HandlerTypePredicate
* HandlerTypePredicate} to group controllers.
* @param prefixes a map with path prefixes as key * @param prefixes a map with path prefixes as key
* @since 5.1 * @since 5.1
* @see org.springframework.web.method.HandlerTypePredicate
*/ */
public void setPathPrefixes(Map<String, HandlerTypePredicate> prefixes) { public void setPathPrefixes(Map<String, Predicate<Class<?>>> prefixes) {
this.pathPrefixes.clear(); this.pathPrefixes.clear();
prefixes.entrySet().stream() prefixes.entrySet().stream()
.filter(entry -> StringUtils.hasText(entry.getKey())) .filter(entry -> StringUtils.hasText(entry.getKey()))
...@@ -80,8 +83,8 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -80,8 +83,8 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
} }
/** /**
* Set the {@link RequestedContentTypeResolver} to use to determine requested media types. * Set the {@link RequestedContentTypeResolver} to use to determine requested
* If not set, the default constructor is used. * media types. If not set, the default constructor is used.
*/ */
public void setContentTypeResolver(RequestedContentTypeResolver contentTypeResolver) { public void setContentTypeResolver(RequestedContentTypeResolver contentTypeResolver) {
Assert.notNull(contentTypeResolver, "'contentTypeResolver' must not be null"); Assert.notNull(contentTypeResolver, "'contentTypeResolver' must not be null");
...@@ -107,7 +110,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -107,7 +110,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
* The configured path prefixes as a read-only, possibly empty map. * The configured path prefixes as a read-only, possibly empty map.
* @since 5.1 * @since 5.1
*/ */
public Map<String, HandlerTypePredicate> getPathPrefixes() { public Map<String, Predicate<Class<?>>> getPathPrefixes() {
return Collections.unmodifiableMap(this.pathPrefixes); return Collections.unmodifiableMap(this.pathPrefixes);
} }
...@@ -144,7 +147,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -144,7 +147,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
if (typeInfo != null) { if (typeInfo != null) {
info = typeInfo.combine(info); info = typeInfo.combine(info);
} }
for (Map.Entry<String, HandlerTypePredicate> entry : this.pathPrefixes.entrySet()) { for (Map.Entry<String, Predicate<Class<?>>> entry : this.pathPrefixes.entrySet()) {
if (entry.getValue().test(handlerType)) { if (entry.getValue().test(handlerType)) {
String prefix = entry.getKey(); String prefix = entry.getKey();
if (this.embeddedValueResolver != null) { if (this.embeddedValueResolver != null) {
......
...@@ -18,10 +18,10 @@ package org.springframework.web.servlet.config.annotation; ...@@ -18,10 +18,10 @@ package org.springframework.web.servlet.config.annotation;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.function.Predicate;
import org.springframework.lang.Nullable; import org.springframework.lang.Nullable;
import org.springframework.util.PathMatcher; import org.springframework.util.PathMatcher;
import org.springframework.web.method.HandlerTypePredicate;
import org.springframework.web.util.UrlPathHelper; import org.springframework.web.util.UrlPathHelper;
/** /**
...@@ -58,7 +58,7 @@ public class PathMatchConfigurer { ...@@ -58,7 +58,7 @@ public class PathMatchConfigurer {
private PathMatcher pathMatcher; private PathMatcher pathMatcher;
@Nullable @Nullable
private Map<String, HandlerTypePredicate> pathPrefixes; private Map<String, Predicate<Class<?>>> pathPrefixes;
/** /**
...@@ -121,13 +121,14 @@ public class PathMatchConfigurer { ...@@ -121,13 +121,14 @@ public class PathMatchConfigurer {
* Configure a path prefix to apply to matching controller methods. * Configure a path prefix to apply to matching controller methods.
* <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping} * <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping}
* method whose controller type is matched by the corresponding * method whose controller type is matched by the corresponding
* {@link HandlerTypePredicate}. The prefix for the first matching predicate * {@code Predicate}. The prefix for the first matching predicate is used.
* is used. * <p>Consider using {@link org.springframework.web.method.HandlerTypePredicate
* HandlerTypePredicate} to group controllers.
* @param prefix the prefix to apply * @param prefix the prefix to apply
* @param predicate a predicate for matching controller types * @param predicate a predicate for matching controller types
* @since 5.1 * @since 5.1
*/ */
public PathMatchConfigurer addPathPrefix(String prefix, HandlerTypePredicate predicate) { public PathMatchConfigurer addPathPrefix(String prefix, Predicate<Class<?>> predicate) {
this.pathPrefixes = this.pathPrefixes == null ? new LinkedHashMap<>() : this.pathPrefixes; this.pathPrefixes = this.pathPrefixes == null ? new LinkedHashMap<>() : this.pathPrefixes;
this.pathPrefixes.put(prefix, predicate); this.pathPrefixes.put(prefix, predicate);
return this; return this;
...@@ -160,7 +161,7 @@ public class PathMatchConfigurer { ...@@ -160,7 +161,7 @@ public class PathMatchConfigurer {
} }
@Nullable @Nullable
protected Map<String, HandlerTypePredicate> getPathPrefixes() { protected Map<String, Predicate<Class<?>>> getPathPrefixes() {
return this.pathPrefixes; return this.pathPrefixes;
} }
} }
...@@ -22,6 +22,7 @@ import java.util.HashMap; ...@@ -22,6 +22,7 @@ import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Locale; import java.util.Locale;
import java.util.Map; import java.util.Map;
import java.util.function.Predicate;
import javax.servlet.ServletContext; import javax.servlet.ServletContext;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
...@@ -70,7 +71,6 @@ import org.springframework.web.bind.WebDataBinder; ...@@ -70,7 +71,6 @@ import org.springframework.web.bind.WebDataBinder;
import org.springframework.web.bind.support.ConfigurableWebBindingInitializer; import org.springframework.web.bind.support.ConfigurableWebBindingInitializer;
import org.springframework.web.context.ServletContextAware; import org.springframework.web.context.ServletContextAware;
import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.method.HandlerTypePredicate;
import org.springframework.web.method.support.CompositeUriComponentsContributor; import org.springframework.web.method.support.CompositeUriComponentsContributor;
import org.springframework.web.method.support.HandlerMethodArgumentResolver; import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.HandlerMethodReturnValueHandler; import org.springframework.web.method.support.HandlerMethodReturnValueHandler;
...@@ -311,7 +311,7 @@ public class WebMvcConfigurationSupport implements ApplicationContextAware, Serv ...@@ -311,7 +311,7 @@ public class WebMvcConfigurationSupport implements ApplicationContextAware, Serv
mapping.setPathMatcher(pathMatcher); mapping.setPathMatcher(pathMatcher);
} }
Map<String, HandlerTypePredicate> pathPrefixes = configurer.getPathPrefixes(); Map<String, Predicate<Class<?>>> pathPrefixes = configurer.getPathPrefixes();
if (pathPrefixes != null) { if (pathPrefixes != null) {
mapping.setPathPrefixes(pathPrefixes); mapping.setPathPrefixes(pathPrefixes);
} }
......
...@@ -23,6 +23,7 @@ import java.util.LinkedHashMap; ...@@ -23,6 +23,7 @@ import java.util.LinkedHashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.function.Predicate;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import org.springframework.context.EmbeddedValueResolverAware; import org.springframework.context.EmbeddedValueResolverAware;
...@@ -38,7 +39,6 @@ import org.springframework.web.bind.annotation.CrossOrigin; ...@@ -38,7 +39,6 @@ import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.method.HandlerTypePredicate;
import org.springframework.web.method.HandlerMethod; import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.handler.MatchableHandlerMapping; import org.springframework.web.servlet.handler.MatchableHandlerMapping;
import org.springframework.web.servlet.handler.RequestMatchResult; import org.springframework.web.servlet.handler.RequestMatchResult;
...@@ -67,7 +67,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -67,7 +67,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
private boolean useTrailingSlashMatch = true; private boolean useTrailingSlashMatch = true;
private final Map<String, HandlerTypePredicate> pathPrefixes = new LinkedHashMap<>(); private final Map<String, Predicate<Class<?>>> pathPrefixes = new LinkedHashMap<>();
private ContentNegotiationManager contentNegotiationManager = new ContentNegotiationManager(); private ContentNegotiationManager contentNegotiationManager = new ContentNegotiationManager();
...@@ -113,12 +113,13 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -113,12 +113,13 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
* Configure path prefixes to apply to controller methods. * Configure path prefixes to apply to controller methods.
* <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping} * <p>Prefixes are used to enrich the mappings of every {@code @RequestMapping}
* method whose controller type is matched by the corresponding * method whose controller type is matched by the corresponding
* {@link HandlerTypePredicate} in the map. The prefix for the first matching * {@code Predicate}. The prefix for the first matching predicate is used.
* predicate is used, assuming the input map has predictable order. * <p>Consider using {@link org.springframework.web.method.HandlerTypePredicate
* HandlerTypePredicate} to group controllers.
* @param prefixes a map with path prefixes as key * @param prefixes a map with path prefixes as key
* @since 5.1 * @since 5.1
*/ */
public void setPathPrefixes(Map<String, HandlerTypePredicate> prefixes) { public void setPathPrefixes(Map<String, Predicate<Class<?>>> prefixes) {
this.pathPrefixes.clear(); this.pathPrefixes.clear();
prefixes.entrySet().stream() prefixes.entrySet().stream()
.filter(entry -> StringUtils.hasText(entry.getKey())) .filter(entry -> StringUtils.hasText(entry.getKey()))
...@@ -178,7 +179,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -178,7 +179,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
* The configured path prefixes as a read-only, possibly empty map. * The configured path prefixes as a read-only, possibly empty map.
* @since 5.1 * @since 5.1
*/ */
public Map<String, HandlerTypePredicate> getPathPrefixes() { public Map<String, Predicate<Class<?>>> getPathPrefixes() {
return Collections.unmodifiableMap(this.pathPrefixes); return Collections.unmodifiableMap(this.pathPrefixes);
} }
...@@ -226,7 +227,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi ...@@ -226,7 +227,7 @@ public class RequestMappingHandlerMapping extends RequestMappingInfoHandlerMappi
if (typeInfo != null) { if (typeInfo != null) {
info = typeInfo.combine(info); info = typeInfo.combine(info);
} }
for (Map.Entry<String, HandlerTypePredicate> entry : this.pathPrefixes.entrySet()) { for (Map.Entry<String, Predicate<Class<?>>> entry : this.pathPrefixes.entrySet()) {
if (entry.getValue().test(handlerType)) { if (entry.getValue().test(handlerType)) {
String prefix = entry.getKey(); String prefix = entry.getKey();
if (this.embeddedValueResolver != null) { if (this.embeddedValueResolver != null) {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册