Polish ServletRegistration API Deferral
Tomcat uses different ServletContext instances from startup- and request-time. This commit ensures that if the programmatic API isn't available at startup-time, then use the ServletContext attached to the HttpServletRequest at runtime. Issue gh-13794
This commit is contained in:
parent
bb0987db80
commit
a98baa7522
|
@ -23,7 +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.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Function;
|
||||||
|
|
||||||
import javax.servlet.DispatcherType;
|
import javax.servlet.DispatcherType;
|
||||||
import javax.servlet.ServletContext;
|
import javax.servlet.ServletContext;
|
||||||
|
@ -44,7 +44,6 @@ import org.springframework.security.web.util.matcher.RegexRequestMatcher;
|
||||||
import org.springframework.security.web.util.matcher.RequestMatcher;
|
import org.springframework.security.web.util.matcher.RequestMatcher;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
import org.springframework.util.ClassUtils;
|
import org.springframework.util.ClassUtils;
|
||||||
import org.springframework.util.function.SingletonSupplier;
|
|
||||||
import org.springframework.web.context.WebApplicationContext;
|
import org.springframework.web.context.WebApplicationContext;
|
||||||
import org.springframework.web.servlet.handler.HandlerMappingIntrospector;
|
import org.springframework.web.servlet.handler.HandlerMappingIntrospector;
|
||||||
|
|
||||||
|
@ -327,7 +326,8 @@ public abstract class AbstractRequestMatcherRegistry<C> {
|
||||||
matchers.add(resolve(ant, mvc, servletContext));
|
matchers.add(resolve(ant, mvc, servletContext));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
matchers.add(new DeferredRequestMatcher(() -> resolve(ant, mvc, servletContext), mvc, ant));
|
matchers.add(new DeferredRequestMatcher((request) -> resolve(ant, mvc, request.getServletContext()),
|
||||||
|
mvc, ant));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return requestMatchers(matchers.toArray(new RequestMatcher[0]));
|
return requestMatchers(matchers.toArray(new RequestMatcher[0]));
|
||||||
|
@ -584,27 +584,34 @@ public abstract class AbstractRequestMatcherRegistry<C> {
|
||||||
|
|
||||||
static class DeferredRequestMatcher implements RequestMatcher {
|
static class DeferredRequestMatcher implements RequestMatcher {
|
||||||
|
|
||||||
final Supplier<RequestMatcher> requestMatcher;
|
final Function<HttpServletRequest, RequestMatcher> requestMatcherFactory;
|
||||||
|
|
||||||
final AtomicReference<String> description = new AtomicReference<>();
|
final AtomicReference<String> description = new AtomicReference<>();
|
||||||
|
|
||||||
DeferredRequestMatcher(Supplier<RequestMatcher> resolver, RequestMatcher... candidates) {
|
volatile RequestMatcher requestMatcher;
|
||||||
this.requestMatcher = SingletonSupplier.of(() -> {
|
|
||||||
RequestMatcher matcher = resolver.get();
|
DeferredRequestMatcher(Function<HttpServletRequest, RequestMatcher> resolver, RequestMatcher... candidates) {
|
||||||
this.description.set(matcher.toString());
|
this.requestMatcherFactory = (request) -> {
|
||||||
return matcher;
|
if (this.requestMatcher == null) {
|
||||||
});
|
synchronized (this) {
|
||||||
|
if (this.requestMatcher == null) {
|
||||||
|
this.requestMatcher = resolver.apply(request);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this.requestMatcher;
|
||||||
|
};
|
||||||
this.description.set("Deferred " + Arrays.toString(candidates));
|
this.description.set("Deferred " + Arrays.toString(candidates));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean matches(HttpServletRequest request) {
|
public boolean matches(HttpServletRequest request) {
|
||||||
return this.requestMatcher.get().matches(request);
|
return this.requestMatcherFactory.apply(request).matches(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public MatchResult matcher(HttpServletRequest request) {
|
public MatchResult matcher(HttpServletRequest request) {
|
||||||
return this.requestMatcher.get().matcher(request);
|
return this.requestMatcherFactory.apply(request).matcher(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -385,7 +385,7 @@ public class AbstractRequestMatcherRegistryTests {
|
||||||
List<RequestMatcher> requestMatchers = new ArrayList<>();
|
List<RequestMatcher> requestMatchers = new ArrayList<>();
|
||||||
for (RequestMatcher requestMatcher : wrappedMatchers) {
|
for (RequestMatcher requestMatcher : wrappedMatchers) {
|
||||||
if (requestMatcher instanceof AbstractRequestMatcherRegistry.DeferredRequestMatcher) {
|
if (requestMatcher instanceof AbstractRequestMatcherRegistry.DeferredRequestMatcher) {
|
||||||
requestMatchers.add(((DeferredRequestMatcher) requestMatcher).requestMatcher.get());
|
requestMatchers.add(((DeferredRequestMatcher) requestMatcher).requestMatcher);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
requestMatchers.add(requestMatcher);
|
requestMatchers.add(requestMatcher);
|
||||||
|
|
Loading…
Reference in New Issue