Integration test each servlet web server with Spring MVC

See gh-46071
This commit is contained in:
Andy Wilkinson 2025-05-08 11:50:47 +01:00
parent d49bf8e59c
commit 3630a43ed7
8 changed files with 166 additions and 71 deletions

View File

@ -50,6 +50,7 @@ dependencies {
testRuntimeOnly("org.eclipse.jetty:jetty-client") testRuntimeOnly("org.eclipse.jetty:jetty-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport")
testRuntimeOnly("org.springframework:spring-webmvc")
} }
test { test {

View File

@ -0,0 +1,45 @@
/*
* Copyright 2012-present 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.boot.jetty.servlet;
import org.springframework.boot.jetty.JettyWebServer;
import org.springframework.boot.web.server.servlet.context.ServletWebServerApplicationContext;
import org.springframework.boot.web.servlet.context.AbstractServletWebServerMvcIntegrationTests;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* Integration tests for {@link ServletWebServerApplicationContext} and
* {@link JettyWebServer} running Spring MVC.
*/
class JettyServletWebServerMvcIntegrationTests extends AbstractServletWebServerMvcIntegrationTests {
protected JettyServletWebServerMvcIntegrationTests() {
super(JettyConfig.class);
}
@Configuration(proxyBeanMethods = false)
static class JettyConfig {
@Bean
JettyServletWebServerFactory webServerFactory() {
return new JettyServletWebServerFactory(0);
}
}
}

View File

@ -58,6 +58,7 @@ dependencies {
testRuntimeOnly("org.eclipse.jetty:jetty-client") testRuntimeOnly("org.eclipse.jetty:jetty-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport")
testRuntimeOnly("org.springframework:spring-webmvc")
tomcatDistribution("org.apache.tomcat:tomcat:${tomcatVersion}@zip") tomcatDistribution("org.apache.tomcat:tomcat:${tomcatVersion}@zip")
} }

View File

@ -0,0 +1,45 @@
/*
* Copyright 2012-present 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.boot.tomcat.servlet;
import org.springframework.boot.tomcat.TomcatWebServer;
import org.springframework.boot.web.server.servlet.context.ServletWebServerApplicationContext;
import org.springframework.boot.web.servlet.context.AbstractServletWebServerMvcIntegrationTests;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* Integration tests for {@link ServletWebServerApplicationContext} and
* {@link TomcatWebServer} running Spring MVC.
*/
class TomcatServletWebServerMvcIntegrationTests extends AbstractServletWebServerMvcIntegrationTests {
protected TomcatServletWebServerMvcIntegrationTests() {
super(TomcatConfig.class);
}
@Configuration(proxyBeanMethods = false)
static class TomcatConfig {
@Bean
TomcatServletWebServerFactory webServerFactory() {
return new TomcatServletWebServerFactory(0);
}
}
}

View File

@ -46,6 +46,7 @@ dependencies {
testRuntimeOnly("org.eclipse.jetty:jetty-client") testRuntimeOnly("org.eclipse.jetty:jetty-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client")
testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport") testRuntimeOnly("org.eclipse.jetty.http2:jetty-http2-client-transport")
testRuntimeOnly("org.springframework:spring-webmvc")
} }
test { test {

View File

@ -0,0 +1,45 @@
/*
* Copyright 2012-present 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.boot.undertow.servlet;
import org.springframework.boot.undertow.UndertowWebServer;
import org.springframework.boot.web.server.servlet.context.ServletWebServerApplicationContext;
import org.springframework.boot.web.servlet.context.AbstractServletWebServerMvcIntegrationTests;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* Integration tests for {@link ServletWebServerApplicationContext} and
* {@link UndertowWebServer} running Spring MVC.
*/
class UndertowServletWebServerMvcIntegrationTests extends AbstractServletWebServerMvcIntegrationTests {
protected UndertowServletWebServerMvcIntegrationTests() {
super(UndertowConfig.class);
}
@Configuration(proxyBeanMethods = false)
static class UndertowConfig {
@Bean
UndertowServletWebServerFactory webServerFactory() {
return new UndertowServletWebServerFactory(0);
}
}
}

View File

@ -48,6 +48,7 @@ dependencies {
testFixturesCompileOnly("org.mockito:mockito-core") testFixturesCompileOnly("org.mockito:mockito-core")
testFixturesCompileOnly("org.springframework:spring-tx") testFixturesCompileOnly("org.springframework:spring-tx")
testFixturesCompileOnly("org.springframework:spring-webflux") testFixturesCompileOnly("org.springframework:spring-webflux")
testFixturesCompileOnly("org.springframework:spring-webmvc")
testImplementation(project(":spring-boot-project:spring-boot-test")) testImplementation(project(":spring-boot-project:spring-boot-test"))
testImplementation(project(":spring-boot-project:spring-boot-tools:spring-boot-test-support")) testImplementation(project(":spring-boot-project:spring-boot-tools:spring-boot-test-support"))

View File

@ -21,19 +21,16 @@ import java.net.URI;
import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import org.springframework.boot.jetty.servlet.JettyServletWebServerFactory;
import org.springframework.boot.testsupport.classpath.resources.WithResource; import org.springframework.boot.testsupport.classpath.resources.WithResource;
import org.springframework.boot.testsupport.web.servlet.DirtiesUrlFactories;
import org.springframework.boot.tomcat.servlet.TomcatServletWebServerFactory;
import org.springframework.boot.undertow.servlet.UndertowServletWebServerFactory;
import org.springframework.boot.web.server.WebServer; import org.springframework.boot.web.server.WebServer;
import org.springframework.boot.web.server.servlet.ServletWebServerFactory; import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.boot.web.server.WebServerFactoryCustomizerBeanPostProcessor;
import org.springframework.boot.web.server.servlet.ConfigurableServletWebServerFactory;
import org.springframework.boot.web.server.servlet.context.AnnotationConfigServletWebServerApplicationContext; import org.springframework.boot.web.server.servlet.context.AnnotationConfigServletWebServerApplicationContext;
import org.springframework.boot.web.server.servlet.context.ServletWebServerApplicationContext; import org.springframework.boot.web.server.servlet.context.ServletWebServerApplicationContext;
import org.springframework.boot.web.servlet.ServletRegistrationBean; import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;
import org.springframework.context.annotation.PropertySource; import org.springframework.context.annotation.PropertySource;
import org.springframework.core.env.Environment; import org.springframework.core.env.Environment;
import org.springframework.http.HttpMethod; import org.springframework.http.HttpMethod;
@ -49,17 +46,22 @@ import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThat;
/** /**
* Integration tests for {@link ServletWebServerApplicationContext} and {@link WebServer}s * Base class for integration testing of {@link ServletWebServerApplicationContext} and
* running Spring MVC. * {@link WebServer}s running Spring MVC.
* *
* @author Phillip Webb * @author Phillip Webb
* @author Ivan Sopov * @author Ivan Sopov
*/ */
@DirtiesUrlFactories public abstract class AbstractServletWebServerMvcIntegrationTests {
class ServletWebServerMvcIntegrationTests {
private AnnotationConfigServletWebServerApplicationContext context; private AnnotationConfigServletWebServerApplicationContext context;
private Class<?> webServerConfiguration;
protected AbstractServletWebServerMvcIntegrationTests(Class<?> webServerConfiguration) {
this.webServerConfiguration = webServerConfiguration;
}
@AfterEach @AfterEach
void closeContext() { void closeContext() {
try { try {
@ -71,27 +73,17 @@ class ServletWebServerMvcIntegrationTests {
} }
@Test @Test
void tomcat() throws Exception { void basicConfig() throws Exception {
this.context = new AnnotationConfigServletWebServerApplicationContext(TomcatConfig.class); this.context = new AnnotationConfigServletWebServerApplicationContext(this.webServerConfiguration,
doTest(this.context, "/hello"); Config.class);
}
@Test
void jetty() throws Exception {
this.context = new AnnotationConfigServletWebServerApplicationContext(JettyConfig.class);
doTest(this.context, "/hello");
}
@Test
void undertow() throws Exception {
this.context = new AnnotationConfigServletWebServerApplicationContext(UndertowConfig.class);
doTest(this.context, "/hello"); doTest(this.context, "/hello");
} }
@Test @Test
@WithResource(name = "conf.properties", content = "context=/example") @WithResource(name = "conf.properties", content = "context=/example")
void advancedConfig() throws Exception { void advancedConfig() throws Exception {
this.context = new AnnotationConfigServletWebServerApplicationContext(AdvancedConfig.class); this.context = new AnnotationConfigServletWebServerApplicationContext(this.webServerConfiguration,
AdvancedConfig.class);
doTest(this.context, "/example/spring/hello"); doTest(this.context, "/example/spring/hello");
} }
@ -105,45 +97,6 @@ class ServletWebServerMvcIntegrationTests {
} }
} }
// Simple main method for testing in a browser
@SuppressWarnings("resource")
static void main(String[] args) {
new AnnotationConfigServletWebServerApplicationContext(JettyServletWebServerFactory.class, Config.class);
}
@Configuration(proxyBeanMethods = false)
@Import(Config.class)
static class TomcatConfig {
@Bean
ServletWebServerFactory webServerFactory() {
return new TomcatServletWebServerFactory(0);
}
}
@Configuration(proxyBeanMethods = false)
@Import(Config.class)
static class JettyConfig {
@Bean
ServletWebServerFactory webServerFactory() {
return new JettyServletWebServerFactory(0);
}
}
@Configuration(proxyBeanMethods = false)
@Import(Config.class)
static class UndertowConfig {
@Bean
ServletWebServerFactory webServerFactory() {
return new UndertowServletWebServerFactory(0);
}
}
@Configuration(proxyBeanMethods = false) @Configuration(proxyBeanMethods = false)
@EnableWebMvc @EnableWebMvc
static class Config { static class Config {
@ -151,9 +104,6 @@ class ServletWebServerMvcIntegrationTests {
@Bean @Bean
DispatcherServlet dispatcherServlet() { DispatcherServlet dispatcherServlet() {
return new DispatcherServlet(); return new DispatcherServlet();
// Alternatively you can use ServletContextInitializer beans including
// ServletRegistration and FilterRegistration. Read the
// EmbeddedWebApplicationContext Javadoc for details.
} }
@Bean @Bean
@ -175,10 +125,16 @@ class ServletWebServerMvcIntegrationTests {
} }
@Bean @Bean
ServletWebServerFactory webServerFactory() { static WebServerFactoryCustomizerBeanPostProcessor webServerFactoryCustomizerBeanPostProcessor() {
JettyServletWebServerFactory factory = new JettyServletWebServerFactory(0); return new WebServerFactoryCustomizerBeanPostProcessor();
factory.setContextPath(this.env.getProperty("context")); }
return factory;
@Bean
WebServerFactoryCustomizer<ConfigurableServletWebServerFactory> contextPathCustomizer() {
return (factory) -> {
String contextPath = this.env.getProperty("context");
factory.setContextPath(contextPath);
};
} }
@Bean @Bean