Close

Spring MVC - @SpringJUnitWebConfig - Spring MVC Testing

[Last Updated: Mar 24, 2026]

@SpringJUnitWebConfig is a composed annotation introduced in Spring 5 that combines @ExtendWith(SpringExtension.class), @ContextConfiguration, and @WebAppConfiguration into a single, concise declaration. It is specifically designed to streamline the setup of Spring MVC integration tests using JUnit 5+

Before this annotation existed, developers had to stack three separate annotations on every test class that needed a full web application context. @SpringJUnitWebConfig removes that boilerplate, making test classes cleaner and easier to read while retaining the full power of the Spring TestContext Framework.

Why Use @SpringJUnitWebConfig?

When writing Spring MVC integration tests, you typically need a WebApplicationContext to instantiate MockMvc. Historically this required combining three annotations manually. @SpringJUnitWebConfig acts as a meta-annotation that packages all three together. Key use cases include:

  • Testing Spring MVC controllers end-to-end without deploying to a server.
  • Verifying request mappings, view resolution, and response payloads in isolation.
  • Reducing annotation clutter on test classes in large projects.

Definition of SpringJUnitWebConfig

Version: 7.0.6
 package org.springframework.test.context.junit.jupiter.web;
 @ExtendWith(SpringExtension.class)
 @ContextConfiguration
 @WebAppConfiguration
 @Documented
 @Inherited
 @Retention(RetentionPolicy.RUNTIME)
 @Target(ElementType.TYPE)
 public @interface SpringJUnitWebConfig {
     @AliasFor(annotation = ContextConfiguration.class, attribute = "classes")
     Class<?>[] value() default {}; 1
     @AliasFor(annotation = ContextConfiguration.class)
     Class<?>[] classes() default {}; 2
     @AliasFor(annotation = ContextConfiguration.class)
     String[] locations() default {}; 3
     @AliasFor(annotation = ContextConfiguration.class)
     Class<? extends ApplicationContextInitializer<?>>[] initializers()
                                                             default {}; 4
     @AliasFor(annotation = ContextConfiguration.class)
     boolean inheritLocations() default true; 5
     @AliasFor(annotation = ContextConfiguration.class)
     boolean inheritInitializers() default true; 6
     @AliasFor(annotation = ContextConfiguration.class)
     Class<? extends ContextLoader> loader()
                                            default ContextLoader.class; 7
     @AliasFor(annotation = ContextConfiguration.class)
     String name() default ""; 8
     @AliasFor(annotation = WebAppConfiguration.class, attribute = "value")
     String resourcePath() default "src/main/webapp"; 9
 }
1Alias for ContextConfiguration#classes.
2Alias for ContextConfiguration#classes.
3Alias for ContextConfiguration#locations.
4Alias for ContextConfiguration#initializers.
5Alias for ContextConfiguration#inheritLocations.
6Alias for ContextConfiguration#inheritInitializers.
7Alias for ContextConfiguration#loader. (Since 6.1)
8Alias for ContextConfiguration#name.
9Alias for WebAppConfiguration#value.

Check out this tutorial to understand @ContextConfiguration annotation.

Example

The following example demonstrates a minimal Spring MVC controller and its corresponding integration test written with @SpringJUnitWebConfig.

package com.logicbig.example;

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class GreetingController {

    @GetMapping("/greet")
    public String greet() {
        return "Hello, World!";
    }
}
package com.logicbig.example;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
@EnableWebMvc
@ComponentScan
public class WebConfig implements WebMvcConfigurer {
}

Integration Test

package com.logicbig.example;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.junit.jupiter.web.SpringJUnitWebConfig;
import org.springframework.test.web.servlet.assertj.MockMvcTester;
import org.springframework.web.context.WebApplicationContext;

import static org.assertj.core.api.Assertions.assertThat;

@SpringJUnitWebConfig(WebConfig.class)
public class GreetingControllerTest {

    @Autowired
    private WebApplicationContext wac;

    private MockMvcTester mockMvc;

    @BeforeEach
    void setup() {
        mockMvc = MockMvcTester.from(wac);
    }

    @Test
    void testGreetEndpoint() {
        assertThat(mockMvc.get().uri("/greet"))
                .hasStatusOk()
                .bodyText().isEqualTo("Hello, World!");

        System.out.println("Greet test passed");
    }
}

Output

$ mvn clean test -Dtest="GreetingControllerTest"
[INFO] Scanning for projects...
[INFO]
[INFO] ---< com.logicbig.example:spring-mvc-testing-using-jUnit-web-config >---
[INFO] Building spring-mvc-testing-using-jUnit-web-config 1.0-SNAPSHOT
[INFO] from pom.xml
[INFO] --------------------------------[ jar ]---------------------------------
[INFO]
[INFO] --- clean:3.2.0:clean (default-clean) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] Deleting D:\example-projects\spring-mvc-testing\spring-mvc-testing-using-jUnit-web-config\target
[INFO]
[INFO] --- resources:3.3.1:resources (default-resources) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] skip non existing resourceDirectory D:\example-projects\spring-mvc-testing\spring-mvc-testing-using-jUnit-web-config\src\main\resources
[INFO]
[INFO] --- compiler:3.15.0:compile (default-compile) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] Recompiling the module because of changed source code.
[INFO] Compiling 2 source files with javac [debug target 25] to target\classes
[INFO]
[INFO] --- resources:3.3.1:testResources (default-testResources) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] skip non existing resourceDirectory D:\example-projects\spring-mvc-testing\spring-mvc-testing-using-jUnit-web-config\src\test\resources
[INFO]
[INFO] --- compiler:3.15.0:testCompile (default-testCompile) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] Recompiling the module because of changed dependency.
[INFO] Compiling 2 source files with javac [debug target 25] to target\test-classes
[INFO]
[INFO] --- surefire:3.5.5:test (default-test) @ spring-mvc-testing-using-jUnit-web-config ---
[INFO] Using auto detected provider org.apache.maven.surefire.junitplatform.JUnitPlatformProvider
[WARNING] file.encoding cannot be set as system property, use <argLine>-Dfile.encoding=...</argLine> instead
[INFO]
[INFO] -------------------------------------------------------
[INFO] T E S T S
[INFO] -------------------------------------------------------
[INFO] Running com.logicbig.example.GreetingControllerTest
Greet test passed
[INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.573 s -- in com.logicbig.example.GreetingControllerTest
[INFO]
[INFO] Results:
[INFO]
[INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 2.535 s
[INFO] Finished at: 2026-03-24T08:59:23+08:00
[INFO] ------------------------------------------------------------------------
Mar 24, 2026 8:59:22 AM org.springframework.mock.web.MockServletContext log
INFO: Initializing Spring TestDispatcherServlet ''
Mar 24, 2026 8:59:22 AM org.springframework.test.web.servlet.TestDispatcherServlet initServletBean
INFO: Initializing Servlet ''
Mar 24, 2026 8:59:22 AM org.springframework.test.web.servlet.TestDispatcherServlet initServletBean
INFO: Completed initialization in 1 ms

Conclusion

The test output confirms that the MockMvc request to /greet receives an HTTP 200 status and the body Hello, World!, exactly as implemented in the controller. This validates that @SpringJUnitWebConfig correctly bootstraps a full WebApplicationContext without any XML or manual annotation stacking, keeping the test class minimal and focused.

Example Project

Dependencies and Technologies Used:

  • spring-webmvc 7.0.6 (Spring Web MVC)
     Version Compatibility: 5.0.0.RELEASE - 7.0.6Version List
    ×

    Version compatibilities of spring-webmvc with this example:

      javax.servlet-api:3.x
    • 5.0.0.RELEASE
    • 5.0.1.RELEASE
    • 5.0.2.RELEASE
    • 5.0.3.RELEASE
    • 5.0.4.RELEASE
    • 5.0.5.RELEASE
    • 5.0.6.RELEASE
    • 5.0.7.RELEASE
    • 5.0.8.RELEASE
    • 5.0.9.RELEASE
    • 5.0.10.RELEASE
    • 5.0.11.RELEASE
    • 5.0.12.RELEASE
    • 5.0.13.RELEASE
    • 5.0.14.RELEASE
    • 5.0.15.RELEASE
    • 5.0.16.RELEASE
    • 5.0.17.RELEASE
    • 5.0.18.RELEASE
    • 5.0.19.RELEASE
    • 5.0.20.RELEASE
    • 5.1.0.RELEASE
    • 5.1.1.RELEASE
    • 5.1.2.RELEASE
    • 5.1.3.RELEASE
    • 5.1.4.RELEASE
    • 5.1.5.RELEASE
    • 5.1.6.RELEASE
    • 5.1.7.RELEASE
    • 5.1.8.RELEASE
    • 5.1.9.RELEASE
    • 5.1.10.RELEASE
    • 5.1.11.RELEASE
    • 5.1.12.RELEASE
    • 5.1.13.RELEASE
    • 5.1.14.RELEASE
    • 5.1.15.RELEASE
    • 5.1.16.RELEASE
    • 5.1.17.RELEASE
    • 5.1.18.RELEASE
    • 5.1.19.RELEASE
    • 5.1.20.RELEASE
    • 5.2.0.RELEASE
    • 5.2.1.RELEASE
    • 5.2.2.RELEASE
    • 5.2.3.RELEASE
    • 5.2.4.RELEASE
    • 5.2.5.RELEASE
    • 5.2.6.RELEASE
    • 5.2.7.RELEASE
    • 5.2.8.RELEASE
    • 5.2.9.RELEASE
    • 5.2.10.RELEASE
    • 5.2.11.RELEASE
    • 5.2.12.RELEASE
    • 5.2.13.RELEASE
    • 5.2.14.RELEASE
    • 5.2.15.RELEASE
    • 5.2.16.RELEASE
    • 5.2.17.RELEASE
    • 5.2.18.RELEASE
    • 5.2.19.RELEASE
    • 5.2.20.RELEASE
    • 5.2.21.RELEASE
    • 5.2.22.RELEASE
    • 5.2.23.RELEASE
    • 5.2.24.RELEASE
    • 5.2.25.RELEASE
    • 5.3.0
    • 5.3.1
    • 5.3.2
    • 5.3.3
    • 5.3.4
    • javax.servlet-api:4.x
    • 5.3.5
    • 5.3.6
    • 5.3.7
    • 5.3.8
    • 5.3.9
    • 5.3.10
    • 5.3.11
    • 5.3.12
    • 5.3.13
    • 5.3.14
    • 5.3.15
    • 5.3.16
    • 5.3.17
    • 5.3.18
    • 5.3.19
    • 5.3.20
    • 5.3.21
    • 5.3.22
    • 5.3.23
    • 5.3.24
    • 5.3.25
    • 5.3.26
    • 5.3.27
    • 5.3.28
    • 5.3.29
    • 5.3.30
    • 5.3.31
    • 5.3.32
    • 5.3.33
    • 5.3.34
    • 5.3.35
    • 5.3.36
    • 5.3.37
    • 5.3.38
    • 5.3.39
    • javax.* -> jakarta.*
      jakarta.servlet-api:6.x
      Java 17 min
    • 6.0.0
    • 6.0.1
    • 6.0.2
    • 6.0.3
    • 6.0.4
    • 6.0.5
    • 6.0.6
    • 6.0.7
    • 6.0.8
    • 6.0.9
    • 6.0.10
    • 6.0.11
    • 6.0.12
    • 6.0.13
    • 6.0.14
    • 6.0.15
    • 6.0.16
    • 6.0.17
    • 6.0.18
    • 6.0.19
    • 6.0.20
    • 6.0.21
    • 6.0.22
    • 6.0.23
    • 6.1.0
    • 6.1.1
    • 6.1.2
    • 6.1.3
    • 6.1.4
    • 6.1.5
    • 6.1.6
    • 6.1.7
    • 6.1.8
    • 6.1.9
    • 6.1.10
    • 6.1.11
    • 6.1.12
    • 6.1.13
    • 6.1.14
    • 6.1.15
    • 6.1.16
    • 6.1.17
    • 6.1.18
    • 6.1.19
    • 6.1.20
    • 6.1.21
    • 6.2.0
    • 6.2.1
    • 6.2.2
    • 6.2.3
    • 6.2.4
    • 6.2.5
    • 6.2.6
    • 6.2.7
    • 6.2.8
    • 6.2.9
    • 6.2.10
    • 6.2.11
    • 6.2.12
    • 6.2.13
    • 6.2.14
    • 6.2.15
    • 6.2.16
    • 6.2.17
    • 7.0.0
    • 7.0.1
    • 7.0.2
    • 7.0.3
    • 7.0.4
    • 7.0.5
    • 7.0.6

    Versions in green have been tested.

  • spring-test 7.0.6 (Spring TestContext Framework)
  • junit-jupiter-engine 6.0.3 (Module "junit-jupiter-engine" of JUnit)
  • jakarta.servlet-api 6.1.0 (Jakarta Servlet API documentation)
  • hamcrest 3.0 (Core API and libraries of hamcrest matcher framework)
  • assertj-core 3.26.3 (Rich and fluent assertions for testing in Java)
  • JDK 25
  • Maven 3.9.11

Spring MVC - Unit Testing using @SpringJUnitWebConfig Select All Download
  • spring-mvc-testing-using-jUnit-web-config
    • src
      • main
        • java
          • com
            • logicbig
              • example
      • test
        • java
          • com
            • logicbig
              • example
                • GreetingControllerTest.java

    See Also

    Join