Add servlet path to /error if it is customized
In addition I added some convenience methods to ServerProperties (servletMapping() and servletPrefix()) for manipulating the servlet path as provided by the user (e.g. normalizing it into a valid Servlet mapping path for the DispatcherServlet). Fixes gh-939, see also gh-936
This commit is contained in:
parent
8acd6fc799
commit
660d9e24dc
|
|
@ -86,7 +86,7 @@ public class DispatcherServletAutoConfiguration {
|
||||||
@Bean(name = DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME)
|
@Bean(name = DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME)
|
||||||
public ServletRegistrationBean dispatcherServletRegistration() {
|
public ServletRegistrationBean dispatcherServletRegistration() {
|
||||||
ServletRegistrationBean registration = new ServletRegistrationBean(
|
ServletRegistrationBean registration = new ServletRegistrationBean(
|
||||||
dispatcherServlet(), this.server.getServletPath());
|
dispatcherServlet(), this.server.getServletMapping());
|
||||||
registration.setName(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME);
|
registration.setName(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME);
|
||||||
if (this.multipartConfig != null) {
|
if (this.multipartConfig != null) {
|
||||||
registration.setMultipartConfig(this.multipartConfig);
|
registration.setMultipartConfig(this.multipartConfig);
|
||||||
|
|
|
||||||
|
|
@ -24,6 +24,7 @@ import javax.servlet.Servlet;
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
import javax.servlet.http.HttpServletResponse;
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.beans.factory.annotation.Value;
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
|
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
|
||||||
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
||||||
|
|
@ -67,11 +68,15 @@ import org.springframework.web.servlet.view.BeanNameViewResolver;
|
||||||
// Ensure this loads before the main WebMvcAutoConfiguration so that the error View is
|
// Ensure this loads before the main WebMvcAutoConfiguration so that the error View is
|
||||||
// available
|
// available
|
||||||
@AutoConfigureBefore(WebMvcAutoConfiguration.class)
|
@AutoConfigureBefore(WebMvcAutoConfiguration.class)
|
||||||
|
@Configuration
|
||||||
public class ErrorMvcAutoConfiguration implements EmbeddedServletContainerCustomizer {
|
public class ErrorMvcAutoConfiguration implements EmbeddedServletContainerCustomizer {
|
||||||
|
|
||||||
@Value("${error.path:/error}")
|
@Value("${error.path:/error}")
|
||||||
private String errorPath = "/error";
|
private String errorPath = "/error";
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private ServerProperties properties;
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@ConditionalOnMissingBean(value = ErrorAttributes.class, search = SearchStrategy.CURRENT)
|
@ConditionalOnMissingBean(value = ErrorAttributes.class, search = SearchStrategy.CURRENT)
|
||||||
public DefaultErrorAttributes errorAttributes() {
|
public DefaultErrorAttributes errorAttributes() {
|
||||||
|
|
@ -86,7 +91,8 @@ public class ErrorMvcAutoConfiguration implements EmbeddedServletContainerCustom
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void customize(ConfigurableEmbeddedServletContainer container) {
|
public void customize(ConfigurableEmbeddedServletContainer container) {
|
||||||
container.addErrorPages(new ErrorPage(this.errorPath));
|
container.addErrorPages(new ErrorPage(this.properties.getServletPrefix()
|
||||||
|
+ this.errorPath));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
|
|
|
||||||
|
|
@ -78,6 +78,33 @@ public class ServerProperties implements EmbeddedServletContainerCustomizer {
|
||||||
return this.servletPath;
|
return this.servletPath;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getServletMapping() {
|
||||||
|
if (this.servletPath.equals("") || this.servletPath.equals("/")) {
|
||||||
|
return "/";
|
||||||
|
}
|
||||||
|
if (this.servletPath.contains("*")) {
|
||||||
|
if (this.servletPath.endsWith("*")) {
|
||||||
|
return this.servletPath;
|
||||||
|
}
|
||||||
|
return this.servletPath;
|
||||||
|
}
|
||||||
|
if (this.servletPath.endsWith("/")) {
|
||||||
|
return this.servletPath + "*";
|
||||||
|
}
|
||||||
|
return this.servletPath + "/*";
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getServletPrefix() {
|
||||||
|
String result = this.servletPath;
|
||||||
|
if (result.contains("*")) {
|
||||||
|
result = result.substring(0, result.indexOf("*"));
|
||||||
|
}
|
||||||
|
if (result.endsWith("/")) {
|
||||||
|
result = result.substring(0, result.length() - 1);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
public void setServletPath(String servletPath) {
|
public void setServletPath(String servletPath) {
|
||||||
this.servletPath = servletPath;
|
this.servletPath = servletPath;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -98,7 +98,7 @@ public class DispatcherServletAutoConfigurationTests {
|
||||||
assertNotNull(this.context.getBean(DispatcherServlet.class));
|
assertNotNull(this.context.getBean(DispatcherServlet.class));
|
||||||
ServletRegistrationBean registration = this.context
|
ServletRegistrationBean registration = this.context
|
||||||
.getBean(ServletRegistrationBean.class);
|
.getBean(ServletRegistrationBean.class);
|
||||||
assertEquals("[/spring]", registration.getUrlMappings().toString());
|
assertEquals("[/spring/*]", registration.getUrlMappings().toString());
|
||||||
assertNull(registration.getMultipartConfig());
|
assertNull(registration.getMultipartConfig());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,100 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2013 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.boot.autoconfigure.web;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
|
import org.springframework.boot.autoconfigure.PropertyPlaceholderAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.web.RemappedErrorViewIntegrationTests.TestConfiguration;
|
||||||
|
import org.springframework.boot.builder.SpringApplicationBuilder;
|
||||||
|
import org.springframework.boot.context.embedded.ConfigurableEmbeddedServletContainer;
|
||||||
|
import org.springframework.boot.context.embedded.EmbeddedServletContainerCustomizer;
|
||||||
|
import org.springframework.boot.context.embedded.ErrorPage;
|
||||||
|
import org.springframework.boot.test.IntegrationTest;
|
||||||
|
import org.springframework.boot.test.SpringApplicationConfiguration;
|
||||||
|
import org.springframework.boot.test.TestRestTemplate;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.Import;
|
||||||
|
import org.springframework.stereotype.Controller;
|
||||||
|
import org.springframework.test.annotation.DirtiesContext;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
import org.springframework.test.context.web.WebAppConfiguration;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.client.RestTemplate;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Dave Syer
|
||||||
|
*/
|
||||||
|
@SpringApplicationConfiguration(classes = TestConfiguration.class)
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@WebAppConfiguration
|
||||||
|
@IntegrationTest({ "server.servletPath:/spring/*", "server.port:0" })
|
||||||
|
@DirtiesContext
|
||||||
|
public class RemappedErrorViewIntegrationTests {
|
||||||
|
|
||||||
|
@Value("${local.server.port}")
|
||||||
|
private int port;
|
||||||
|
|
||||||
|
private RestTemplate template = new TestRestTemplate();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void directAccessToErrorPage() throws Exception {
|
||||||
|
String content = this.template.getForObject("http://localhost:" + this.port
|
||||||
|
+ "/spring/error", String.class);
|
||||||
|
assertTrue("Wrong content: " + content, content.contains("error"));
|
||||||
|
assertTrue("Wrong content: " + content, content.contains("999"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void forwardToErrorPage() throws Exception {
|
||||||
|
String content = this.template.getForObject("http://localhost:" + this.port
|
||||||
|
+ "/spring/", String.class);
|
||||||
|
assertTrue("Wrong content: " + content, content.contains("error"));
|
||||||
|
assertTrue("Wrong content: " + content, content.contains("500"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@Import({ PropertyPlaceholderAutoConfiguration.class,
|
||||||
|
ServerPropertiesAutoConfiguration.class, WebMvcAutoConfiguration.class,
|
||||||
|
HttpMessageConvertersAutoConfiguration.class,
|
||||||
|
EmbeddedServletContainerAutoConfiguration.class,
|
||||||
|
DispatcherServletAutoConfiguration.class, ErrorMvcAutoConfiguration.class })
|
||||||
|
@Controller
|
||||||
|
public static class TestConfiguration implements EmbeddedServletContainerCustomizer {
|
||||||
|
|
||||||
|
@RequestMapping("/")
|
||||||
|
public String home() {
|
||||||
|
throw new RuntimeException("Planned!");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void customize(ConfigurableEmbeddedServletContainer container) {
|
||||||
|
container.addErrorPages(new ErrorPage("/spring/error"));
|
||||||
|
}
|
||||||
|
|
||||||
|
// For manual testing
|
||||||
|
public static void main(String[] args) {
|
||||||
|
new SpringApplicationBuilder(TestConfiguration.class).properties(
|
||||||
|
"server.servletPath:spring/*").run(args);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue