Spring Boot Reference 3.1.x
Spring Boot Reference 3.1.x
Phillip Webb, Dave Syer, Josh Long, Stéphane Nicoll, Rob Winch, Andy
Wilkinson, Marcel Overdijk, Christian Dupuis, Sébastien Deleuze, Michael
Simons, Vedran Pavić, Jay Bryant, Madhura Bhave, Eddú Meléndez, Scott
Frederick, Moritz Halbritter
Version 3.1.0-RC2
Table of Contents
1. Legal . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 2
2. Getting Help . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3
3. Documentation Overview . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3.1. First Steps . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3.2. Upgrading From an Earlier Version . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3.3. Developing With Spring Boot. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3.4. Learning About Spring Boot Features . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3.5. Web . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.6. Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.7. Messaging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.8. IO . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.9. Container Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
3.10. Moving to Production . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
3.11. GraalVM Native Images. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
3.12. Advanced Topics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
4. Getting Started . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
4.1. Introducing Spring Boot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
4.2. System Requirements . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
4.2.1. Servlet Containers. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
4.2.2. GraalVM Native Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
4.3. Installing Spring Boot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
4.3.1. Installation Instructions for the Java Developer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
Maven Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
Gradle Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
4.3.2. Installing the Spring Boot CLI . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
Manual Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
Installation with SDKMAN!. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
OSX Homebrew Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
MacPorts Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
Command-line Completion . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
Windows Scoop Installation. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
4.4. Developing Your First Spring Boot Application. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
4.4.1. Prerequisites . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
4.4.2. Setting up the project with Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
4.4.3. Setting up the project with Gradle . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
4.4.4. Adding Classpath Dependencies . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
4.4.5. Writing the Code . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 17
The @RestController and @RequestMapping Annotations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 18
The @SpringBootApplication Annotation. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 18
The “main” Method . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 18
4.4.6. Running the Example. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 18
Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
4.4.7. Creating an Executable Jar . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20
Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20
Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21
4.5. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 22
5. Upgrading Spring Boot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
5.1. Upgrading From 1.x . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
5.2. Upgrading to a New Feature Release . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
5.3. Upgrading the Spring Boot CLI . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
5.4. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
6. Developing with Spring Boot. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
6.1. Build Systems. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
6.1.1. Dependency Management . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
6.1.2. Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
6.1.3. Gradle . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
6.1.4. Ant . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 26
6.1.5. Starters . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 27
6.2. Structuring Your Code . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 31
6.2.1. Using the “default” Package . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 31
6.2.2. Locating the Main Application Class . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 31
6.3. Configuration Classes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
6.3.1. Importing Additional Configuration Classes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
6.3.2. Importing XML Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
6.4. Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 33
6.4.1. Gradually Replacing Auto-configuration. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 34
6.4.2. Disabling Specific Auto-configuration Classes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 34
6.5. Spring Beans and Dependency Injection . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 35
6.6. Using the @SpringBootApplication Annotation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37
6.7. Running Your Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 40
6.7.1. Running From an IDE . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 40
6.7.2. Running as a Packaged Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 40
6.7.3. Using the Maven Plugin. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 40
6.7.4. Using the Gradle Plugin . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 41
6.7.5. Hot Swapping. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 41
6.8. Developer Tools. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 41
6.8.1. Diagnosing Classloading Issues . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 42
6.8.2. Property Defaults . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 42
6.8.3. Automatic Restart . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 43
Logging Changes in Condition Evaluation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 45
Excluding Resources . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 45
Watching Additional Paths . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 46
Disabling Restart . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 46
Using a Trigger File. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 47
Customizing the Restart Classloader . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 48
Known Limitations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 49
6.8.4. LiveReload . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 49
6.8.5. Global Settings . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 49
Configuring File System Watcher . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 50
6.8.6. Remote Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 51
Running the Remote Client Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 51
Remote Update. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 52
6.9. Packaging Your Application for Production. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 53
6.10. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 53
7. Core Features. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 54
7.1. SpringApplication. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 54
7.1.1. Startup Failure. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 55
7.1.2. Lazy Initialization. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 56
7.1.3. Customizing the Banner . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 57
7.1.4. Customizing SpringApplication. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 58
7.1.5. Fluent Builder API . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 59
7.1.6. Application Availability. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
Liveness State. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
Readiness State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
Managing the Application Availability State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
7.1.7. Application Events and Listeners . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 64
7.1.8. Web Environment. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 65
7.1.9. Accessing Application Arguments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 65
7.1.10. Using the ApplicationRunner or CommandLineRunner. . . . . . . . . . . . . . . . . . . . . . . . . . . . . 66
7.1.11. Application Exit. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 67
7.1.12. Admin Features . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 69
7.1.13. Application Startup tracking . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 69
7.2. Externalized Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 70
7.2.1. Accessing Command Line Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 72
7.2.2. JSON Application Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 73
7.2.3. External Application Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 73
Optional Locations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 75
Wildcard Locations. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 76
Profile Specific Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 76
Importing Additional Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 77
Importing Extensionless Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 79
Using Configuration Trees . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 79
Property Placeholders . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 82
Working With Multi-Document Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 82
Activation Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 83
7.2.4. Encrypting Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 84
7.2.5. Working With YAML . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 84
Mapping YAML to Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 84
Directly Loading YAML . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 85
7.2.6. Configuring Random Values. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 86
7.2.7. Configuring System Environment Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 86
7.2.8. Type-safe Configuration Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 86
JavaBean Properties Binding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 87
Constructor Binding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 90
Enabling @ConfigurationProperties-annotated Types . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 93
Using @ConfigurationProperties-annotated Types . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 95
Third-party Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 97
Relaxed Binding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 98
Merging Complex Types . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 101
Properties Conversion . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 105
@ConfigurationProperties Validation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 112
@ConfigurationProperties vs. @Value. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 115
7.3. Profiles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 116
7.3.1. Adding Active Profiles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 118
7.3.2. Profile Groups . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 119
7.3.3. Programmatically Setting Profiles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 120
7.3.4. Profile-specific Configuration Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 120
7.4. Logging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 120
7.4.1. Log Format . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 120
7.4.2. Console Output . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 121
Color-coded Output . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 122
7.4.3. File Output . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 123
7.4.4. File Rotation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 123
7.4.5. Log Levels . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 124
7.4.6. Log Groups . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 124
7.4.7. Using a Log Shutdown Hook . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 125
7.4.8. Custom Log Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 126
7.4.9. Logback Extensions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 128
Profile-specific Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 128
Environment Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 129
7.4.10. Log4j2 Extensions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 129
Profile-specific Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 130
Environment Properties Lookup . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 130
Log4j2 System Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 130
7.5. Internationalization. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 131
7.6. JSON . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 132
7.6.1. Jackson . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 132
Custom Serializers and Deserializers. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 132
Mixins . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 136
7.6.2. Gson . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 136
7.6.3. JSON-B . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 137
7.7. Task Execution and Scheduling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 137
7.8. Testing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 138
7.8.1. Test Scope Dependencies . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 139
7.8.2. Testing Spring Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 139
7.8.3. Testing Spring Boot Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 139
Detecting Web Application Type . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 140
Detecting Test Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 141
Using the Test Configuration Main Method . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 142
Excluding Test Configuration. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 144
Using Application Arguments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 145
Testing With a Mock Environment. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 146
Testing With a Running Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 150
Customizing WebTestClient . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 152
Using JMX . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 152
Using Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 154
Using Tracing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 154
Mocking and Spying Beans. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 154
Auto-configured Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 158
Auto-configured JSON Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 158
Auto-configured Spring MVC Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 161
Auto-configured Spring WebFlux Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 165
Auto-configured Spring GraphQL Tests. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 168
Auto-configured Data Cassandra Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 173
Auto-configured Data Couchbase Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 174
Auto-configured Data Elasticsearch Tests. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 175
Auto-configured Data JPA Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 176
Auto-configured JDBC Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 179
Auto-configured Data JDBC Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 180
Auto-configured jOOQ Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 180
Auto-configured Data MongoDB Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 181
Auto-configured Data Neo4j Tests. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 182
Auto-configured Data Redis Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 184
Auto-configured Data LDAP Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 185
Auto-configured REST Clients . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 186
Auto-configured Spring REST Docs Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 188
Auto-configured Spring Web Services Tests . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 199
Additional Auto-configuration and Slicing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 203
User Configuration and Slicing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 204
Using Spock to Test Spring Boot Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 208
7.8.4. Testcontainers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 208
Service Connections . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 210
Dynamic Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 212
Using Testcontainers at Development Time. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 214
7.8.5. Test Utilities . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 220
ConfigDataApplicationContextInitializer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 220
TestPropertyValues. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 220
OutputCapture . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 221
TestRestTemplate . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 222
7.9. Docker Compose Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 226
7.9.1. Service Connections . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 227
7.9.2. Custom Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 227
7.9.3. Skipping Specific Containers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 228
7.9.4. Using a Specific Compose File . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 228
7.9.5. Waiting for Container Readiness . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 229
7.9.6. Controlling the Docker Compose Lifecycle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 230
7.9.7. Activating Docker Compose Profiles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 230
7.10. Creating Your Own Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 231
7.10.1. Understanding Auto-configured Beans . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 231
7.10.2. Locating Auto-configuration Candidates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 231
7.10.3. Condition Annotations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 232
Class Conditions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 232
Bean Conditions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 234
Property Conditions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 236
Resource Conditions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 236
Web Application Conditions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 236
SpEL Expression Conditions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 236
7.10.4. Testing your Auto-configuration. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 236
Simulating a Web Context. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 239
Overriding the Classpath . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 240
7.10.5. Creating Your Own Starter . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 240
Naming . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 241
Configuration keys . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 241
The “autoconfigure” Module . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 243
Starter Module. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 244
7.11. Kotlin Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 245
7.11.1. Requirements . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 245
7.11.2. Null-safety . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 245
7.11.3. Kotlin API . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 246
runApplication . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 246
Extensions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 246
7.11.4. Dependency management . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 247
7.11.5. @ConfigurationProperties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 247
7.11.6. Testing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 247
7.11.7. Resources . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 248
Further reading. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 248
Examples. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 248
7.12. SSL . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 248
7.12.1. Configuring SSL With Java KeyStore Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 248
7.12.2. Configuring SSL With PEM-encoded Certificates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 249
7.12.3. Applying SSL Bundles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 250
7.12.4. Using SSL Bundles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 251
7.13. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 252
8. Web . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 253
8.1. Servlet Web Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 253
8.1.1. The “Spring Web MVC Framework” . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 253
Spring MVC Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 259
HttpMessageConverters . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 259
MessageCodesResolver . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 261
Static Content . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 261
Welcome Page . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 263
Custom Favicon . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 263
Path Matching and Content Negotiation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 263
ConfigurableWebBindingInitializer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 265
Template Engines . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 265
Error Handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 266
CORS Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 275
8.1.2. JAX-RS and Jersey . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 276
8.1.3. Embedded Servlet Container Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 278
Servlets, Filters, and Listeners. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 278
Servlet Context Initialization . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 278
The ServletWebServerApplicationContext . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 279
Customizing Embedded Servlet Containers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 280
JSP Limitations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 284
8.2. Reactive Web Applications. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 284
8.2.1. The “Spring WebFlux Framework” . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 285
Spring WebFlux Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 291
HTTP Codecs with HttpMessageReaders and HttpMessageWriters . . . . . . . . . . . . . . . . . . . . . . 291
Static Content . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 292
Welcome Page . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 293
Template Engines . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 293
Error Handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 294
Web Filters . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 299
8.2.2. Embedded Reactive Server Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 299
8.2.3. Reactive Server Resources Configuration. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 299
8.3. Graceful Shutdown . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 300
8.4. Spring Security . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 301
8.4.1. MVC Security . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 301
8.4.2. WebFlux Security . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 302
8.4.3. OAuth2. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 304
Client . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 304
Resource Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 309
Authorization Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 310
8.4.4. SAML 2.0 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 313
Relying Party . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 313
8.5. Spring Session . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 316
8.6. Spring for GraphQL . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 317
8.6.1. GraphQL Schema . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 318
8.6.2. GraphQL RuntimeWiring . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 319
8.6.3. Querydsl and QueryByExample Repositories Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 320
8.6.4. Transports . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 321
HTTP and WebSocket . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 321
RSocket . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 321
8.6.5. Exception Handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 323
8.6.6. GraphiQL and Schema printer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 323
8.7. Spring HATEOAS . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 323
8.8. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 323
9. Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 324
9.1. SQL Databases . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 324
9.1.1. Configure a DataSource. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 324
Embedded Database Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 324
Connection to a Production Database . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 325
DataSource Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 325
Supported Connection Pools . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 326
Connection to a JNDI DataSource . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 327
9.1.2. Using JdbcTemplate . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 327
9.1.3. JPA and Spring Data JPA . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 329
Entity Classes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 329
Spring Data JPA Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 331
Spring Data Envers Repositories. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 333
Creating and Dropping JPA Databases. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 333
Open EntityManager in View. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 334
9.1.4. Spring Data JDBC. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 334
9.1.5. Using H2’s Web Console . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 335
Changing the H2 Console’s Path . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 335
Accessing the H2 Console in a Secured Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 335
9.1.6. Using jOOQ . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 337
Code Generation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 337
Using DSLContext . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 338
jOOQ SQL Dialect . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 340
Customizing jOOQ. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 340
9.1.7. Using R2DBC. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 340
Embedded Database Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 343
Using DatabaseClient . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 344
Spring Data R2DBC Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 344
9.2. Working with NoSQL Technologies . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 345
9.2.1. Redis. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 346
Connecting to Redis . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 346
9.2.2. MongoDB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 348
Connecting to a MongoDB Database . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 348
MongoTemplate. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 352
Spring Data MongoDB Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 353
9.2.3. Neo4j . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 354
Connecting to a Neo4j Database . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 354
Spring Data Neo4j Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 356
9.2.4. Elasticsearch . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 359
Connecting to Elasticsearch Using REST clients . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 359
Connecting to Elasticsearch by Using Spring Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 360
Spring Data Elasticsearch Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 361
9.2.5. Cassandra . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 362
Connecting to Cassandra. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 362
Spring Data Cassandra Repositories. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 365
9.2.6. Couchbase. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 365
Connecting to Couchbase . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 366
Spring Data Couchbase Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 366
9.2.7. LDAP . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 369
Connecting to an LDAP Server. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 369
Spring Data LDAP Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 370
Embedded In-memory LDAP Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 371
9.2.8. InfluxDB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 372
Connecting to InfluxDB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 372
9.3. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 372
10. Messaging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 373
10.1. JMS . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 373
10.1.1. ActiveMQ Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 373
10.1.2. ActiveMQ Artemis Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 374
10.1.3. Using a JNDI ConnectionFactory . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 376
10.1.4. Sending a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 376
10.1.5. Receiving a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 377
10.2. AMQP. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 381
10.2.1. RabbitMQ Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 381
10.2.2. Sending a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 382
10.2.3. Sending a Message To A Stream . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 384
10.2.4. Receiving a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 385
10.3. Apache Kafka Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 388
10.3.1. Sending a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 389
10.3.2. Receiving a Message . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 390
10.3.3. Kafka Streams . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 391
10.3.4. Additional Kafka Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 392
10.3.5. Testing with Embedded Kafka. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 394
10.4. RSocket . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 396
10.4.1. RSocket Strategies Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 396
10.4.2. RSocket server Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 396
10.4.3. Spring Messaging RSocket support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 397
10.4.4. Calling RSocket Services with RSocketRequester . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 397
10.5. Spring Integration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 399
10.6. WebSockets . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 401
10.7. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 401
11. IO . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 402
11.1. Caching . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 402
11.1.1. Supported Cache Providers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 403
Generic . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 405
JCache (JSR-107). . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 405
Hazelcast . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 406
Infinispan . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 406
Couchbase. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 407
Redis. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 409
Caffeine . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 411
Cache2k . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 412
Simple . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 413
None . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 413
11.2. Hazelcast . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 414
11.3. Quartz Scheduler . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 415
11.4. Sending Email . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 418
11.5. Validation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 419
11.6. Calling REST Services . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 420
11.6.1. RestTemplate . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 421
RestTemplate Customization . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 422
RestTemplate SSL Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 426
11.6.2. WebClient . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 427
WebClient Runtime . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 428
WebClient Customization . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 429
WebClient SSL Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 429
11.7. Web Services . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 431
11.7.1. Calling Web Services with WebServiceTemplate . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 432
11.8. Distributed Transactions With JTA . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 435
11.8.1. Using a Jakarta EE Managed Transaction Manager . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 435
11.8.2. Mixing XA and Non-XA JMS Connections . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 435
11.8.3. Supporting an Embedded Transaction Manager . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 436
11.9. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 436
12. Container Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 437
12.1. Efficient Container Images. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 437
12.1.1. Layering Docker Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 437
12.2. Dockerfiles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 438
12.3. Cloud Native Buildpacks. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 439
12.4. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 440
13. Production-ready Features . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 441
13.1. Enabling Production-ready Features. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 441
13.2. Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 441
13.2.1. Enabling Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 443
13.2.2. Exposing Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 444
13.2.3. Security . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 445
Cross Site Request Forgery Protection . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 448
13.2.4. Configuring Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 448
13.2.5. Hypermedia for Actuator Web Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 449
13.2.6. CORS Support. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 449
13.2.7. Implementing Custom Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 450
Receiving Input . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 451
Custom Web Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 452
Servlet Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 453
Controller Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 453
13.2.8. Health Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 454
Auto-configured HealthIndicators . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 454
Writing Custom HealthIndicators. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 455
Reactive Health Indicators . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 458
Auto-configured ReactiveHealthIndicators . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 460
Health Groups . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 460
DataSource Health . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 462
13.2.9. Kubernetes Probes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 462
Checking External State With Kubernetes Probes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 463
Application Lifecycle and Probe States . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 465
13.2.10. Application Information. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 465
Auto-configured InfoContributors . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 466
Custom Application Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 466
Git Commit Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 467
Build Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 468
Java Information. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 468
OS Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 468
Writing Custom InfoContributors. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 468
13.3. Monitoring and Management Over HTTP . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 469
13.3.1. Customizing the Management Endpoint Paths . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 470
13.3.2. Customizing the Management Server Port. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 471
13.3.3. Configuring Management-specific SSL . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 471
13.3.4. Customizing the Management Server Address . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 473
13.3.5. Disabling HTTP Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 473
13.4. Monitoring and Management over JMX . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 474
13.4.1. Customizing MBean Names . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 474
13.4.2. Disabling JMX Endpoints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 475
13.5. Observability . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 475
13.6. Loggers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 476
13.6.1. Configure a Logger . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 477
13.7. Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 477
13.7.1. Getting started. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 478
13.7.2. Supported Monitoring Systems . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 481
AppOptics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 481
Atlas . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 482
Datadog . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 482
Dynatrace . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 483
Elastic. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 486
Ganglia. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 487
Graphite . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 487
Humio . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 489
Influx . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 490
JMX . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 490
KairosDB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 492
New Relic . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 492
OpenTelemetry . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 494
Prometheus . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 494
SignalFx. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 495
Simple . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 495
Stackdriver . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 496
StatsD . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 496
Wavefront. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 497
13.7.3. Supported Metrics and Meters . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 498
JVM Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 498
System Metrics. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 499
Application Startup Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 499
Logger Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 499
Task Execution and Scheduling Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 499
Spring MVC Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 499
Spring WebFlux Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 500
Jersey Server Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 500
HTTP Client Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 501
Tomcat Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 501
Cache Metrics. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 501
Spring Batch Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 502
Spring GraphQL Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 502
DataSource Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 502
Hibernate Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 502
Spring Data Repository Metrics. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 503
RabbitMQ Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 503
Spring Integration Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 503
Kafka Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 504
MongoDB Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 504
Jetty Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 507
@Timed Annotation Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 507
Redis Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 507
13.7.4. Registering Custom Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 507
13.7.5. Customizing Individual Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 509
Common Tags. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 510
Per-meter Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 511
13.7.6. Metrics Endpoint . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 512
13.7.7. Integration with Micrometer Observation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 512
13.8. Tracing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 512
13.8.1. Supported Tracers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 512
13.8.2. Getting Started. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 513
13.8.3. Tracer Implementations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 514
OpenTelemetry With Zipkin. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 514
OpenTelemetry With Wavefront. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 515
OpenTelemetry With OTLP. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 515
OpenZipkin Brave With Zipkin . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 515
OpenZipkin Brave With Wavefront . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 515
13.8.4. Integration with Micrometer Observation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 515
13.8.5. Creating Custom Spans . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 516
13.9. Auditing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 516
13.9.1. Custom Auditing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 517
13.10. Recording HTTP Exchanges. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 517
13.10.1. Custom HTTP Exchange Recording . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 517
13.11. Process Monitoring . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 517
13.11.1. Extending Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 517
13.11.2. Programmatically Enabling Process Monitoring . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 518
13.12. Cloud Foundry Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 518
13.12.1. Disabling Extended Cloud Foundry Actuator Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . 518
13.12.2. Cloud Foundry Self-signed Certificates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 518
13.12.3. Custom Context Path . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 519
13.13. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 522
14. Deploying Spring Boot Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 523
14.1. Deploying to the Cloud . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 523
14.1.1. Cloud Foundry. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 523
Binding to Services . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 525
14.1.2. Kubernetes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 526
Kubernetes Container Lifecycle . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 526
14.1.3. Heroku . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 527
14.1.4. OpenShift . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 528
14.1.5. Amazon Web Services (AWS). . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 529
AWS Elastic Beanstalk . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 529
Summary. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 530
14.1.6. CloudCaptain and Amazon Web Services. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 530
14.1.7. Azure . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 531
14.1.8. Google Cloud . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 531
14.2. Installing Spring Boot Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 532
14.2.1. Supported Operating Systems . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 533
14.2.2. Unix/Linux Services . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 533
Installation as an init.d Service (System V). . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 533
Installation as a systemd Service . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 535
Customizing the Startup Script . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 536
14.2.3. Microsoft Windows Services . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 539
14.3. Efficient deployments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 539
14.3.1. Unpacking the Executable JAR . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 539
14.3.2. Using Ahead-of-time Processing With the JVM . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 540
14.4. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 541
15. GraalVM Native Image Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 542
15.1. Introducing GraalVM Native Images. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 542
15.1.1. Key Differences with JVM Deployments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 542
15.1.2. Understanding Spring Ahead-of-Time Processing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 543
Source Code Generation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 543
Hint File Generation. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 546
Proxy Class Generation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 546
15.2. Developing Your First GraalVM Native Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 547
15.2.1. Sample Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 547
15.2.2. Building a Native Image Using Buildpacks. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 548
System Requirements . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 548
Using Maven. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 548
Using Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 549
Running the example. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 549
15.2.3. Building a Native Image using Native Build Tools . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 550
Prerequisites . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 550
Using Maven. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 550
Using Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 551
Running the Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 551
15.3. Testing GraalVM Native Images . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 552
15.3.1. Testing Ahead-of-time Processing With the JVM. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 552
15.3.2. Testing With Native Build Tools . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 553
Using Maven. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 553
Using Gradle. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 554
15.4. Advanced Native Images Topics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 554
15.4.1. Nested Configuration Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 554
15.4.2. Converting a Spring Boot Executable Jar . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 556
Using Buildpacks. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 556
Using GraalVM native-image . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 557
15.4.3. Using the Tracing Agent . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 558
Launch the Application Directly . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 558
15.4.4. Custom Hints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 558
Testing custom hints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 559
15.4.5. Known Limitations. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 560
15.5. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 560
16. Spring Boot CLI. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 561
16.1. Installing the CLI . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 561
16.2. Using the CLI . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 561
16.2.1. Initialize a New Project . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 563
16.2.2. Using the Embedded Shell . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 564
17. Build Tool Plugins . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 565
17.1. Spring Boot Maven Plugin . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 565
17.2. Spring Boot Gradle Plugin . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 565
17.3. Spring Boot AntLib Module . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 565
17.3.1. Spring Boot Ant Tasks . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 566
Using the “exejar” Task . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 566
Examples. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 566
17.3.2. Using the “findmainclass” Task. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 567
Examples. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 567
17.4. Supporting Other Build Systems. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 567
17.4.1. Repackaging Archives . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 568
17.4.2. Nested Libraries . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 568
17.4.3. Finding a Main Class . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 568
17.4.4. Example Repackage Implementation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 568
17.5. What to Read Next . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 570
18. “How-to” Guides . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 571
18.1. Spring Boot Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 571
18.1.1. Create Your Own FailureAnalyzer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 571
18.1.2. Troubleshoot Auto-configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 571
18.1.3. Customize the Environment or ApplicationContext Before It Starts . . . . . . . . . . . . . . . . . 572
18.1.4. Build an ApplicationContext Hierarchy (Adding a Parent or Root Context) . . . . . . . . . . 575
18.1.5. Create a Non-web Application. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 575
18.2. Properties and Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 575
18.2.1. Automatically Expand Properties at Build Time. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 575
Automatic Property Expansion Using Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 575
Automatic Property Expansion Using Gradle . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 576
18.2.2. Externalize the Configuration of SpringApplication . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 577
18.2.3. Change the Location of External Properties of an Application . . . . . . . . . . . . . . . . . . . . . . 580
18.2.4. Use ‘Short’ Command Line Arguments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 580
18.2.5. Use YAML for External Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 581
18.2.6. Set the Active Spring Profiles. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 581
18.2.7. Set the Default Profile Name . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 582
18.2.8. Change Configuration Depending on the Environment . . . . . . . . . . . . . . . . . . . . . . . . . . . . 582
18.2.9. Discover Built-in Options for External Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 583
18.3. Embedded Web Servers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 584
18.3.1. Use Another Web Server. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 584
18.3.2. Disabling the Web Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 586
18.3.3. Change the HTTP Port . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 586
18.3.4. Use a Random Unassigned HTTP Port . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 586
18.3.5. Discover the HTTP Port at Runtime . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 586
18.3.6. Enable HTTP Response Compression . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 587
18.3.7. Configure SSL . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 588
18.3.8. Configure HTTP/2 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 589
HTTP/2 With Tomcat . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
HTTP/2 With Jetty . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
HTTP/2 With Reactor Netty. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
HTTP/2 With Undertow . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
18.3.9. Configure the Web Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 590
18.3.10. Add a Servlet, Filter, or Listener to an Application . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 592
Add a Servlet, Filter, or Listener by Using a Spring Bean . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 592
Add Servlets, Filters, and Listeners by Using Classpath Scanning . . . . . . . . . . . . . . . . . . . . . . . 593
18.3.11. Configure Access Logging. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 593
18.3.12. Running Behind a Front-end Proxy Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 595
Customize Tomcat’s Proxy Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 595
18.3.13. Enable Multiple Connectors with Tomcat . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 596
18.3.14. Enable Tomcat’s MBean Registry . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 598
18.3.15. Enable Multiple Listeners with Undertow . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 599
18.3.16. Create WebSocket Endpoints Using @ServerEndpoint . . . . . . . . . . . . . . . . . . . . . . . . . . . . 600
18.4. Spring MVC . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 601
18.4.1. Write a JSON REST Service . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 601
18.4.2. Write an XML REST Service . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 602
18.4.3. Customize the Jackson ObjectMapper . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 603
18.4.4. Customize the @ResponseBody Rendering . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 605
18.4.5. Handling Multipart File Uploads . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 605
18.4.6. Switch Off the Spring MVC DispatcherServlet. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 606
18.4.7. Switch off the Default MVC Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 606
18.4.8. Customize ViewResolvers. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 606
18.5. Jersey. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 607
18.5.1. Secure Jersey endpoints with Spring Security. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
18.5.2. Use Jersey Alongside Another Web Framework . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 608
18.6. HTTP Clients . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 609
18.6.1. Configure RestTemplate to Use a Proxy . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 609
18.6.2. Configure the TcpClient used by a Reactor Netty-based WebClient . . . . . . . . . . . . . . . . . . 609
18.7. Logging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 611
18.7.1. Configure Logback for Logging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 612
Configure Logback for File-only Output . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 613
18.7.2. Configure Log4j for Logging . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 614
Use YAML or JSON to Configure Log4j 2 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 615
Use Composite Configuration to Configure Log4j 2 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 616
18.8. Data Access . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 616
18.8.1. Configure a Custom DataSource . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 616
18.8.2. Configure Two DataSources. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 623
18.8.3. Use Spring Data Repositories. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 629
18.8.4. Separate @Entity Definitions from Spring Configuration . . . . . . . . . . . . . . . . . . . . . . . . . . 629
18.8.5. Configure JPA Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 630
18.8.6. Configure Hibernate Naming Strategy . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 631
18.8.7. Configure Hibernate Second-Level Caching . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 633
18.8.8. Use Dependency Injection in Hibernate Components. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 634
18.8.9. Use a Custom EntityManagerFactory. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 635
18.8.10. Using Multiple EntityManagerFactories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 635
18.8.11. Use a Traditional persistence.xml File . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 639
18.8.12. Use Spring Data JPA and Mongo Repositories . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 639
18.8.13. Customize Spring Data’s Web Support . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 640
18.8.14. Expose Spring Data Repositories as REST Endpoint . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 640
18.8.15. Configure a Component that is Used by JPA . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 640
18.8.16. Configure jOOQ with Two DataSources . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 641
18.9. Database Initialization . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 641
18.9.1. Initialize a Database Using JPA . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 642
18.9.2. Initialize a Database Using Hibernate . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 642
18.9.3. Initialize a Database Using Basic SQL Scripts . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 642
18.9.4. Initialize a Spring Batch Database . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 643
18.9.5. Use a Higher-level Database Migration Tool . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 643
Execute Flyway Database Migrations on Startup . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 643
Execute Liquibase Database Migrations on Startup. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 645
18.9.6. Depend Upon an Initialized Database . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 646
Detect a Database Initializer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 646
Detect a Bean That Depends On Database Initialization. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 646
18.10. NoSQL . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 646
18.10.1. Use Jedis Instead of Lettuce . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 647
18.11. Messaging. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 647
18.11.1. Disable Transacted JMS Session . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 647
18.12. Batch Applications . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 649
18.12.1. Specifying a Batch Data Source. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 649
18.12.2. Running Spring Batch Jobs on Startup . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 649
18.12.3. Running From the Command Line. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 650
18.12.4. Storing the Job Repository . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 650
18.13. Actuator . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 650
18.13.1. Change the HTTP Port or Address of the Actuator Endpoints . . . . . . . . . . . . . . . . . . . . . . 650
18.13.2. Customize the ‘whitelabel’ Error Page . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 651
18.13.3. Sanitize Sensitive Values . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 651
Customizing Sanitization . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 652
18.13.4. Map Health Indicators to Micrometer Metrics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 652
18.14. Security. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 654
18.14.1. Switch off the Spring Boot Security Configuration. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 654
18.14.2. Change the UserDetailsService and Add User Accounts . . . . . . . . . . . . . . . . . . . . . . . . . . . 655
18.14.3. Enable HTTPS When Running behind a Proxy Server . . . . . . . . . . . . . . . . . . . . . . . . . . . . 655
18.15. Hot Swapping . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 656
18.15.1. Reload Static Content. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 656
18.15.2. Reload Templates without Restarting the Container . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
Thymeleaf Templates. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
FreeMarker Templates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
Groovy Templates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
18.15.3. Fast Application Restarts . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
18.15.4. Reload Java Classes without Restarting the Container . . . . . . . . . . . . . . . . . . . . . . . . . . . . 657
18.16. Testing. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 658
18.16.1. Testing With Spring Security . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 658
18.16.2. Structure @Configuration classes for inclusion in slice tests . . . . . . . . . . . . . . . . . . . . . . . 659
18.17. Build . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 661
18.17.1. Generate Build Information. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 662
18.17.2. Generate Git Information . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 662
18.17.3. Customize Dependency Versions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 663
18.17.4. Create an Executable JAR with Maven . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 663
18.17.5. Use a Spring Boot Application as a Dependency. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 664
18.17.6. Extract Specific Libraries When an Executable Jar Runs . . . . . . . . . . . . . . . . . . . . . . . . . . 665
18.17.7. Create a Non-executable JAR with Exclusions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 666
18.17.8. Remote Debug a Spring Boot Application Started with Maven. . . . . . . . . . . . . . . . . . . . . 666
18.17.9. Build an Executable Archive From Ant without Using spring-boot-antlib. . . . . . . . . . . 666
18.18. Traditional Deployment . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 667
18.18.1. Create a Deployable War File . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 668
18.18.2. Convert an Existing Application to Spring Boot . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 670
18.18.3. Deploying a WAR to WebLogic . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 675
Appendices. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 677
Appendix A: Common Application Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 677
.A.1. Core Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 677
.A.2. Cache Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 685
.A.3. Mail Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 686
.A.4. JSON Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 687
.A.5. Data Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 689
.A.6. Transaction Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 713
.A.7. Data Migration Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 715
.A.8. Integration Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 722
.A.9. Web Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 745
.A.10. Templating Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 755
.A.11. Server Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 761
.A.12. Security Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 774
.A.13. RSocket Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 776
.A.14. Actuator Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 778
.A.15. Devtools Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 808
.A.16. Docker Compose Properties. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 809
.A.17. Testing Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 810
Appendix B: Configuration Metadata . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 810
.B.1. Metadata Format . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 810
Group Attributes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 812
Property Attributes. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 813
Hint Attributes. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 815
Repeated Metadata Items . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 816
.B.2. Providing Manual Hints . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 817
Value Hint . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 817
Value Providers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 818
.B.3. Generating Your Own Metadata by Using the Annotation Processor . . . . . . . . . . . . . . . . . . 824
Configuring the Annotation Processor . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 824
Automatic Metadata Generation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 825
Adding Additional Metadata . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 831
Appendix C: Auto-configuration Classes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 831
.C.1. spring-boot-autoconfigure . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 831
.C.2. spring-boot-actuator-autoconfigure. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 836
Appendix D: Test Auto-configuration Annotations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 839
.D.1. Test Slices . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 839
Appendix E: The Executable Jar Format . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 848
.E.1. Nested JARs . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 848
The Executable Jar File Structure . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 848
The Executable War File Structure. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 849
Index Files . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 850
Classpath Index . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 850
Layer Index . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 850
.E.2. Spring Boot’s “JarFile” Class. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 851
Compatibility With the Standard Java “JarFile” . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 851
.E.3. Launching Executable Jars. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 851
Launcher Manifest . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 852
.E.4. PropertiesLauncher Features . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 852
.E.5. Executable Jar Restrictions. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 854
.E.6. Alternative Single Jar Solutions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 854
Appendix F: Dependency Versions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 854
.F.1. Managed Dependency Coordinates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 854
.F.2. Version Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 892
This document is also available as multiple HTML pages and as a single HTML
page.
1
Chapter 1. Legal
Copyright © 2012-2023
Copies of this document may be made for your own use and for distribution to others, provided
that you do not charge any fee for such copies and further provided that each copy contains this
Copyright Notice, whether distributed in print or electronically.
2
Chapter 2. Getting Help
If you have trouble with Spring Boot, we would like to help.
• Try the How-to documents. They provide solutions to the most common questions.
• Learn the Spring basics. Spring Boot builds on many other Spring projects. Check the spring.io
web-site for a wealth of reference documentation. If you are starting out with Spring, try one of
the guides.
All of Spring Boot is open source, including the documentation. If you find problems
NOTE
with the docs or if you want to improve them, please get involved.
3
Chapter 3. Documentation Overview
This section provides a brief overview of Spring Boot reference documentation. It serves as a map
for the rest of the document.
Depending on the version that you are upgrading to, you can find some additional tips here:
• Profiles: Profiles
4
• Logging: Logging
3.5. Web
If you develop Spring Boot web applications, take a look at the following content:
3.6. Data
If your application deals with a datastore, you can see how to configure that here:
• SQL: Configuring a SQL Datastore, Embedded Database support, Connection pools, and more.
• NOSQL: Auto-configuration for NOSQL stores such as Redis, MongoDB, Neo4j, and others.
3.7. Messaging
If your application uses any messaging protocol, see one or more of the following sections:
• JMS: Auto-configuration for ActiveMQ and Artemis, Sending and Receiving messages through
JMS
3.8. IO
If your application needs IO capabilities, see one or more of the following sections:
5
• JTA: Distributed Transactions with JTA
• Efficient Container Images: Tips to optimize container images such as Docker images
• Cloud Native Buildpacks: Support for Cloud Native Buildpacks with Maven and Gradle
• GraalVM Native Images: Introduction | Key Differences with the JVM | Ahead-of-Time
Processing
6
Chapter 4. Getting Started
If you are getting started with Spring Boot, or “Spring” in general, start by reading this section. It
answers the basic “what?”, “how?” and “why?” questions. It includes an introduction to Spring
Boot, along with installation instructions. We then walk you through building your first Spring Boot
application, discussing some core principles as we go.
You can use Spring Boot to create Java applications that can be started by using java -jar or more
traditional war deployments.
• Provide a radically faster and widely accessible getting-started experience for all Spring
development.
• Be opinionated out of the box but get out of the way quickly as requirements start to diverge
from the defaults.
• Provide a range of non-functional features that are common to large classes of projects (such as
embedded servers, security, metrics, health checks, and externalized configuration).
• Absolutely no code generation (when not targeting native image) and no requirement for XML
configuration.
7
Name Servlet Version
You can also deploy Spring Boot applications to any servlet 5.0+ compatible container.
Spring Boot applications can be converted into a Native Image using GraalVM 22.3 or above.
Images can be created using the native build tools Gradle/Maven plugins or native-image tool
provided by GraalVM. You can also create native images using the native-image Paketo buildpack.
Name Version
$ java -version
If you are new to Java development or if you want to experiment with Spring Boot, you might want
to try the Spring Boot CLI (Command Line Interface) first. Otherwise, read on for “classic”
installation instructions.
You can use Spring Boot in the same way as any standard Java library. To do so, include the
appropriate spring-boot-*.jar files on your classpath. Spring Boot does not require any special
tools integration, so you can use any IDE or text editor. Also, there is nothing special about a Spring
Boot application, so you can run and debug a Spring Boot application as you would any other Java
program.
Although you could copy Spring Boot jars, we generally recommend that you use a build tool that
supports dependency management (such as Maven or Gradle).
Maven Installation
Spring Boot is compatible with Apache Maven 3.6.3 or later. If you do not already have Maven
installed, you can follow the instructions at maven.apache.org.
8
On many operating systems, Maven can be installed with a package manager. If you
use OSX Homebrew, try brew install maven. Ubuntu users can run sudo apt-get
TIP
install maven. Windows users with Chocolatey can run choco install maven from an
elevated (administrator) prompt.
Spring Boot dependencies use the org.springframework.boot group id. Typically, your Maven POM
file inherits from the spring-boot-starter-parent project and declares dependencies to one or more
“Starters”. Spring Boot also provides an optional Maven plugin to create executable jars.
More details on getting started with Spring Boot and Maven can be found in the Getting Started
section of the Maven plugin’s reference guide.
Gradle Installation
Spring Boot is compatible with Gradle 7.x (7.5 or later) and 8.x. If you do not already have Gradle
installed, you can follow the instructions at gradle.org.
Spring Boot dependencies can be declared by using the org.springframework.boot group. Typically,
your project declares dependencies to one or more “Starters”. Spring Boot provides a useful Gradle
plugin that can be used to simplify dependency declarations and to create executable jars.
Gradle Wrapper
The Gradle Wrapper provides a nice way of “obtaining” Gradle when you need to build a
project. It is a small script and library that you commit alongside your code to bootstrap the
build process. See docs.gradle.org/current/userguide/gradle_wrapper.html for details.
More details on getting started with Spring Boot and Gradle can be found in the Getting Started
section of the Gradle plugin’s reference guide.
The Spring Boot CLI (Command Line Interface) is a command line tool that you can use to quickly
prototype with Spring.
You do not need to use the CLI to work with Spring Boot, but it is a quick way to get a Spring
application off the ground without an IDE.
Manual Installation
You can download the Spring CLI distribution from one of the following locations:
• spring-boot-cli-3.1.0-RC2-bin.zip
• spring-boot-cli-3.1.0-RC2-bin.tar.gz
Once downloaded, follow the INSTALL.txt instructions from the unpacked archive. In summary,
there is a spring script (spring.bat for Windows) in a bin/ directory in the .zip file. Alternatively,
you can use java -jar with the .jar file (the script helps you to be sure that the classpath is set
9
correctly).
SDKMAN! (The Software Development Kit Manager) can be used for managing multiple versions of
various binary SDKs, including Groovy and the Spring Boot CLI. Get SDKMAN! from sdkman.io and
install Spring Boot by using the following commands:
If you develop features for the CLI and want access to the version you built, use the following
commands:
The preceding instructions install a local instance of spring called the dev instance. It points at your
target build location, so every time you rebuild Spring Boot, spring is up-to-date.
$ sdk ls springboot
================================================================================
Available Springboot Versions
================================================================================
> + dev
* 3.1.0-RC2
================================================================================
+ - local version
* - installed
> - currently in use
================================================================================
If you are on a Mac and use Homebrew, you can install the Spring Boot CLI by using the following
commands:
10
$ brew tap spring-io/tap
$ brew install spring-boot
If you do not see the formula, your installation of brew might be out-of-date. In that
NOTE
case, run brew update and try again.
MacPorts Installation
If you are on a Mac and use MacPorts, you can install the Spring Boot CLI by using the following
command:
Command-line Completion
The Spring Boot CLI includes scripts that provide command completion for the BASH and zsh shells.
You can source the script (also named spring) in any shell or put it in your personal or system-wide
bash completion initialization. On a Debian system, the system-wide scripts are in <installation
location>/shell-completion/bash and all scripts in that directory are executed when a new shell
starts. For example, to run the script manually if you have installed by using SDKMAN!, use the
following commands:
$ . ~/.sdkman/candidates/springboot/current/shell-completion/bash/spring
$ spring <HIT TAB HERE>
grab help jar run test version
If you install the Spring Boot CLI by using Homebrew or MacPorts, the command-
NOTE
line completion scripts are automatically registered with your shell.
If you are on a Windows and use Scoop, you can install the Spring Boot CLI by using the following
commands:
If you do not see the app manifest, your installation of scoop might be out-of-date.
NOTE
In that case, run scoop update and try again.
11
4.4. Developing Your First Spring Boot Application
This section describes how to develop a small “Hello World!” web application that highlights some
of Spring Boot’s key features. You can choose between Maven or Gradle as the build system.
The spring.io website contains many “Getting Started” guides that use Spring Boot. If
you need to solve a specific problem, check there first.
TIP You can shortcut the steps below by going to start.spring.io and choosing the "Web"
starter from the dependencies searcher. Doing so generates a new project structure so
that you can start coding right away. Check the start.spring.io user guide for more
details.
4.4.1. Prerequisites
Before we begin, open a terminal and run the following commands to ensure that you have a valid
version of Java installed:
$ java -version
openjdk version "17.0.4.1" 2022-08-12 LTS
OpenJDK Runtime Environment (build 17.0.4.1+1-LTS)
OpenJDK 64-Bit Server VM (build 17.0.4.1+1-LTS, mixed mode, sharing)
Maven
If you want to use Maven, ensure that you have Maven installed:
$ mvn -v
Apache Maven 3.8.5 (3599d3414f046de2324203b78ddcf9b5e4388aa0)
Maven home: usr/Users/developer/tools/maven/3.8.5
Java version: 17.0.4.1, vendor: BellSoft, runtime:
/Users/developer/sdkman/candidates/java/17.0.4.1-librca
Gradle
If you want to use Gradle, ensure that you have Gradle installed:
12
$ gradle --version
------------------------------------------------------------
Gradle 8.1.1
------------------------------------------------------------
Kotlin: 1.8.10
Groovy: 3.0.15
Ant: Apache Ant(TM) version 1.10.11 compiled on July 10 2021
JVM: 17.0.7 (BellSoft 17.0.7+7-LTS)
OS: Linux 6.2.12-200.fc37.aarch64 aarch64
We need to start by creating a Maven pom.xml file. The pom.xml is the recipe that is used to build your
project. Open your favorite text editor and add the following:
13
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>myproject</artifactId>
<version>0.0.1-SNAPSHOT</version>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>3.1.0-RC2</version>
</parent>
<!-- (you only need this if you are using a milestone or snapshot version) -->
<repositories>
<repository>
<id>spring-snapshots</id>
<url>https://repo.spring.io/snapshot</url>
<snapshots><enabled>true</enabled></snapshots>
</repository>
<repository>
<id>spring-milestones</id>
<url>https://repo.spring.io/milestone</url>
</repository>
</repositories>
<pluginRepositories>
<pluginRepository>
<id>spring-snapshots</id>
<url>https://repo.spring.io/snapshot</url>
</pluginRepository>
<pluginRepository>
<id>spring-milestones</id>
<url>https://repo.spring.io/milestone</url>
</pluginRepository>
</pluginRepositories>
</project>
The preceding listing should give you a working build. You can test it by running mvn package (for
now, you can ignore the “jar will be empty - no content was marked for inclusion!” warning).
At this point, you could import the project into an IDE (most modern Java IDEs
NOTE include built-in support for Maven). For simplicity, we continue to use a plain text
editor for this example.
14
4.4.3. Setting up the project with Gradle
We need to start by creating a Gradle build.gradle file. The build.gradle is the build script that is
used to build your project. Open your favorite text editor and add the following:
plugins {
id 'java'
id 'org.springframework.boot' version '3.1.0-RC2'
id 'io.spring.dependency-management' version '1.1.0'
}
group = 'com.example'
version = '0.0.1-SNAPSHOT'
sourceCompatibility = '17'
repositories {
mavenCentral()
maven { url 'https://repo.spring.io/milestone' }
maven { url 'https://repo.spring.io/snapshot' }
}
dependencies {
}
The preceding listing should give you a working build. You can test it by running gradle classes.
At this point, you could import the project into an IDE (most modern Java IDEs
NOTE include built-in support for Gradle). For simplicity, we continue to use a plain text
editor for this example.
Spring Boot provides a number of “Starters” that let you add jars to your classpath. “Starters”
provide dependencies that you are likely to need when developing a specific type of application.
Maven
Most Spring Boot applications use the spring-boot-starter-parent in the parent section of the POM.
The spring-boot-starter-parent is a special starter that provides useful Maven defaults. It also
provides a dependency-management section so that you can omit version tags for “blessed”
dependencies.
15
$ mvn dependency:tree
[INFO] com.example:myproject:jar:0.0.1-SNAPSHOT
The mvn dependency:tree command prints a tree representation of your project dependencies. You
can see that spring-boot-starter-parent provides no dependencies by itself. To add the necessary
dependencies, edit your pom.xml and add the spring-boot-starter-web dependency immediately
below the parent section:
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>
If you run mvn dependency:tree again, you see that there are now a number of additional
dependencies, including the Tomcat web server and Spring Boot itself.
Gradle
Most Spring Boot applications use the org.springframework.boot Gradle plugin. This plugin provides
useful defaults and Gradle tasks. The io.spring.dependency-management Gradle plugin provides
dependency management so that you can omit version tags for “blessed” dependencies.
$ gradle dependencies
------------------------------------------------------------
Root project 'myproject'
------------------------------------------------------------
The gradle dependencies command prints a tree representation of your project dependencies. Right
now, the project has no dependencies. To add the necessary dependencies, edit your build.gradle
and add the spring-boot-starter-web dependency in the dependencies section:
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-web'
}
If you run gradle dependencies again, you see that there are now a number of additional
16
dependencies, including the Tomcat web server and Spring Boot itself.
To finish our application, we need to create a single Java file. By default, Maven and Gradle
compiles sources from src/main/java, so you need to create that directory structure and then add a
file named src/main/java/MyApplication.java to contain the following code:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@SpringBootApplication
public class MyApplication {
@RequestMapping("/")
String home() {
return "Hello World!";
}
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController
@RestController
@SpringBootApplication
class MyApplication {
@RequestMapping("/")
fun home() = "Hello World!"
17
Although there is not much code here, quite a lot is going on. We step through the important parts
in the next few sections.
The first annotation on our MyApplication class is @RestController. This is known as a stereotype
annotation. It provides hints for people reading the code and for Spring that the class plays a
specific role. In this case, our class is a web @Controller, so Spring considers it when handling
incoming web requests.
The @RequestMapping annotation provides “routing” information. It tells Spring that any HTTP
request with the / path should be mapped to the home method. The @RestController annotation tells
Spring to render the resulting string directly back to the caller.
Auto-configuration is designed to work well with “Starters”, but the two concepts are not
directly tied. You are free to pick and choose jar dependencies outside of the starters. Spring
Boot still does its best to auto-configure your application.
The final part of our application is the main method. This is a standard method that follows the Java
convention for an application entry point. Our main method delegates to Spring Boot’s
SpringApplication class by calling run. SpringApplication bootstraps our application, starting Spring,
which, in turn, starts the auto-configured Tomcat web server. We need to pass MyApplication.class
as an argument to the run method to tell SpringApplication which is the primary Spring component.
The args array is also passed through to expose any command-line arguments.
18
Maven
At this point, your application should work. Since you used the spring-boot-starter-parent POM,
you have a useful run goal that you can use to start the application. Type mvn spring-boot:run from
the root project directory to start the application. You should see output similar to the following:
$ mvn spring-boot:run
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.906 seconds (process running for 6.514)
If you open a web browser to localhost:8080, you should see the following output:
Hello World!
Gradle
At this point, your application should work. Since you used the org.springframework.boot Gradle
plugin, you have a useful bootRun goal that you can use to start the application. Type gradle bootRun
from the root project directory to start the application. You should see output similar to the
following:
$ gradle bootRun
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.906 seconds (process running for 6.514)
19
If you open a web browser to localhost:8080, you should see the following output:
Hello World!
We finish our example by creating a completely self-contained executable jar file that we could run
in production. Executable jars (sometimes called “fat jars”) are archives containing your compiled
classes along with all of the jar dependencies that your code needs to run.
Java does not provide a standard way to load nested jar files (jar files that are themselves
contained within a jar). This can be problematic if you are looking to distribute a self-
contained application.
To solve this problem, many developers use “uber” jars. An uber jar packages all the classes
from all the application’s dependencies into a single archive. The problem with this approach
is that it becomes hard to see which libraries are in your application. It can also be
problematic if the same filename is used (but with different content) in multiple jars.
Spring Boot takes a different approach and lets you actually nest jars directly.
Maven
To create an executable jar, we need to add the spring-boot-maven-plugin to our pom.xml. To do so,
insert the following lines just below the dependencies section:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
Save your pom.xml and run mvn package from the command line, as follows:
20
$ mvn package
If you look in the target directory, you should see myproject-0.0.1-SNAPSHOT.jar. The file should be
around 18 MB in size. If you want to peek inside, you can use jar tvf, as follows:
You should also see a much smaller file named myproject-0.0.1-SNAPSHOT.jar.original in the target
directory. This is the original jar file that Maven created before it was repackaged by Spring Boot.
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.999 seconds (process running for 1.253)
Gradle
To create an executable jar, we need to run gradle bootJar from the command line, as follows:
21
$ gradle bootJar
If you look in the build/libs directory, you should see myproject-0.0.1-SNAPSHOT.jar. The file should
be around 18 MB in size. If you want to peek inside, you can use jar tvf, as follows:
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.999 seconds (process running for 1.253)
Otherwise, the next logical step is to read Developing with Spring Boot. If you are really impatient,
you could also jump ahead and read about Spring Boot features.
22
Chapter 5. Upgrading Spring Boot
Instructions for how to upgrade from earlier versions of Spring Boot are provided on the project
wiki. Follow the links in the release notes section to find the version that you want to upgrade to.
Upgrading instructions are always the first item in the release notes. If you are more than one
release behind, please make sure that you also review the release notes of the versions that you
jumped.
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-properties-migrator</artifactId>
<scope>runtime</scope>
</dependency>
Properties that are added late to the environment, such as when using
WARNING
@PropertySource, will not be taken into account.
Once you finish the migration, please make sure to remove this module from your
NOTE
project’s dependencies.
23
Spring Boot’s documentation is specific to that version, so any information that you find in here
will contain the most up-to-date changes that are in that version.
24
Chapter 6. Developing with Spring Boot
This section goes into more detail about how you should use Spring Boot. It covers topics such as
build systems, auto-configuration, and how to run your applications. We also cover some Spring
Boot best practices. Although there is nothing particularly special about Spring Boot (it is just
another library that you can consume), there are a few recommendations that, when followed,
make your development process a little easier.
If you are starting out with Spring Boot, you should probably read the Getting Started guide before
diving into this section.
Each release of Spring Boot provides a curated list of dependencies that it supports. In practice, you
do not need to provide a version for any of these dependencies in your build configuration, as
Spring Boot manages that for you. When you upgrade Spring Boot itself, these dependencies are
upgraded as well in a consistent way.
You can still specify a version and override Spring Boot’s recommendations if you
NOTE
need to do so.
The curated list contains all the Spring modules that you can use with Spring Boot as well as a
refined list of third party libraries. The list is available as a standard Bills of Materials (spring-boot-
dependencies) that can be used with both Maven and Gradle.
Each release of Spring Boot is associated with a base version of the Spring
WARNING
Framework. We highly recommend that you do not specify its version.
6.1.2. Maven
To learn about using Spring Boot with Maven, see the documentation for Spring Boot’s Maven
plugin:
• API
6.1.3. Gradle
To learn about using Spring Boot with Gradle, see the documentation for Spring Boot’s Gradle
plugin:
25
• Reference (HTML and PDF)
• API
6.1.4. Ant
It is possible to build a Spring Boot project using Apache Ant+Ivy. The spring-boot-antlib “AntLib”
module is also available to help Ant create executable jars.
To declare dependencies, a typical ivy.xml file looks something like the following example:
<ivy-module version="2.0">
<info organisation="org.springframework.boot" module="spring-boot-sample-ant" />
<configurations>
<conf name="compile" description="everything needed to compile this module" />
<conf name="runtime" extends="compile" description="everything needed to run
this module" />
</configurations>
<dependencies>
<dependency org="org.springframework.boot" name="spring-boot-starter"
rev="${spring-boot.version}" conf="compile" />
</dependencies>
</ivy-module>
26
<project
xmlns:ivy="antlib:org.apache.ivy.ant"
xmlns:spring-boot="antlib:org.springframework.boot.ant"
name="myapp" default="build">
If you do not want to use the spring-boot-antlib module, see the Build an Executable
TIP
Archive From Ant without Using spring-boot-antlib “How-to” .
6.1.5. Starters
Starters are a set of convenient dependency descriptors that you can include in your application.
You get a one-stop shop for all the Spring and related technologies that you need without having to
hunt through sample code and copy-paste loads of dependency descriptors. For example, if you
want to get started using Spring and JPA for database access, include the spring-boot-starter-data-
jpa dependency in your project.
The starters contain a lot of the dependencies that you need to get a project up and running quickly
and with a consistent, supported set of managed transitive dependencies.
27
What is in a name
As explained in the “Creating Your Own Starter” section, third party starters should not start
with spring-boot, as it is reserved for official Spring Boot artifacts. Rather, a third-party
starter typically starts with the name of the project. For example, a third-party starter project
called thirdpartyproject would typically be named thirdpartyproject-spring-boot-starter.
The following application starters are provided by Spring Boot under the org.springframework.boot
group:
Name Description
28
Name Description
29
Name Description
In addition to the application starters, the following starters can be used to add production ready
features:
Name Description
Finally, Spring Boot also includes the following starters that can be used if you want to exclude or
swap specific technical facets:
Name Description
30
Name Description
To learn how to swap technical facets, please see the how-to documentation for swapping web
server and logging system.
For a list of additional community contributed starters, see the README file in the
TIP
spring-boot-starters module on GitHub.
When a class does not include a package declaration, it is considered to be in the “default package”.
The use of the “default package” is generally discouraged and should be avoided. It can cause
particular problems for Spring Boot applications that use the @ComponentScan,
@ConfigurationPropertiesScan, @EntityScan, or @SpringBootApplication annotations, since every class
from every jar is read.
We generally recommend that you locate your main application class in a root package above other
classes. The @SpringBootApplication annotation is often placed on your main class, and it implicitly
defines a base “search package” for certain items. For example, if you are writing a JPA application,
the package of the @SpringBootApplication annotated class is used to search for @Entity items. Using
a root package also allows component scan to apply only on your project.
31
If you do not want to use @SpringBootApplication, the @EnableAutoConfiguration and
TIP @ComponentScan annotations that it imports defines that behavior so you can also use
those instead.
com
+- example
+- myapplication
+- MyApplication.java
|
+- customer
| +- Customer.java
| +- CustomerController.java
| +- CustomerService.java
| +- CustomerRepository.java
|
+- order
+- Order.java
+- OrderController.java
+- OrderService.java
+- OrderRepository.java
The MyApplication.java file would declare the main method, along with the basic
@SpringBootApplication, as follows:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
32
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
Many Spring configuration examples have been published on the Internet that use
TIP XML configuration. If possible, always try to use the equivalent Java-based
configuration. Searching for Enable* annotations can be a good starting point.
You need not put all your @Configuration into a single class. The @Import annotation can be used to
import additional configuration classes. Alternatively, you can use @ComponentScan to automatically
pick up all Spring components, including @Configuration classes.
If you absolutely must use XML based configuration, we recommend that you still start with a
@Configuration class. You can then use an @ImportResource annotation to load XML configuration
files.
6.4. Auto-configuration
Spring Boot auto-configuration attempts to automatically configure your Spring application based
on the jar dependencies that you have added. For example, if HSQLDB is on your classpath, and you
have not manually configured any database connection beans, then Spring Boot auto-configures an
in-memory database.
33
6.4.1. Gradually Replacing Auto-configuration
Auto-configuration is non-invasive. At any point, you can start to define your own configuration to
replace specific parts of the auto-configuration. For example, if you add your own DataSource bean,
the default embedded database support backs away.
If you need to find out what auto-configuration is currently being applied, and why, start your
application with the --debug switch. Doing so enables debug logs for a selection of core loggers and
logs a conditions report to the console.
If you find that specific auto-configuration classes that you do not want are being applied, you can
use the exclude attribute of @SpringBootApplication to disable them, as shown in the following
example:
Java
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
@SpringBootApplication(exclude = { DataSourceAutoConfiguration.class })
public class MyApplication {
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration
@SpringBootApplication(exclude = [DataSourceAutoConfiguration::class])
class MyApplication
If the class is not on the classpath, you can use the excludeName attribute of the annotation and
specify the fully qualified name instead. If you prefer to use @EnableAutoConfiguration rather than
@SpringBootApplication, exclude and excludeName are also available. Finally, you can also control the
list of auto-configuration classes to exclude by using the spring.autoconfigure.exclude property.
TIP You can define exclusions both at the annotation level and by using the property.
Even though auto-configuration classes are public, the only aspect of the class that
is considered public API is the name of the class which can be used for disabling the
NOTE auto-configuration. The actual contents of those classes, such as nested
configuration classes or bean methods are for internal use only and we do not
recommend using those directly.
34
6.5. Spring Beans and Dependency Injection
You are free to use any of the standard Spring Framework techniques to define your beans and
their injected dependencies. We generally recommend using constructor injection to wire up
dependencies and @ComponentScan to find beans.
If you structure your code as suggested above (locating your application class in a top package), you
can add @ComponentScan without any arguments or use the @SpringBootApplication annotation which
implicitly includes it. All of your application components (@Component, @Service, @Repository,
@Controller, and others) are automatically registered as Spring Beans.
The following example shows a @Service Bean that uses constructor injection to obtain a required
RiskAssessor bean:
Java
import org.springframework.stereotype.Service;
@Service
public class MyAccountService implements AccountService {
// ...
Kotlin
import org.springframework.stereotype.Service
@Service
class MyAccountService(private val riskAssessor: RiskAssessor) : AccountService
If a bean has more than one constructor, you will need to mark the one you want Spring to use with
@Autowired:
35
Java
import java.io.PrintStream;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class MyAccountService implements AccountService {
@Autowired
public MyAccountService(RiskAssessor riskAssessor) {
this.riskAssessor = riskAssessor;
this.out = System.out;
}
// ...
36
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.stereotype.Service
import java.io.PrintStream
@Service
class MyAccountService : AccountService {
@Autowired
constructor(riskAssessor: RiskAssessor) {
this.riskAssessor = riskAssessor
out = System.out
}
// ...
Notice how using constructor injection lets the riskAssessor field be marked as final,
TIP
indicating that it cannot be subsequently changed.
• @ComponentScan: enable @Component scan on the package where the application is located (see the
best practices)
37
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
38
None of these features are mandatory and you may choose to replace this single
annotation by any of the features that it enables. For instance, you may not want to
use component scan or configuration properties scan in your application:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.SpringBootConfiguration;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.context.annotation.Import;
@SpringBootConfiguration(proxyBeanMethods = false)
@EnableAutoConfiguration
@Import({ SomeConfiguration.class, AnotherConfiguration.class })
public class MyApplication {
NOTE Kotlin
import org.springframework.boot.SpringBootConfiguration
import org.springframework.boot.autoconfigure.EnableAutoConfiguration
import
org.springframework.boot.docs.using.structuringyourcode.locatingthemainc
lass.MyApplication
import org.springframework.boot.runApplication
import org.springframework.context.annotation.Import
@SpringBootConfiguration(proxyBeanMethods = false)
@EnableAutoConfiguration
@Import(SomeConfiguration::class, AnotherConfiguration::class)
class MyApplication
In this example, MyApplication is just like any other Spring Boot application except
that @Component-annotated classes and @ConfigurationProperties-annotated classes
are not detected automatically and the user-defined beans are imported explicitly
(see @Import).
39
6.7. Running Your Application
One of the biggest advantages of packaging your application as a jar and using an embedded HTTP
server is that you can run your application as you would any other. The sample applies to
debugging Spring Boot applications. You do not need any special IDE plugins or extensions.
This section only covers jar-based packaging. If you choose to package your
NOTE
application as a war file, see your server and IDE documentation.
You can run a Spring Boot application from your IDE as a Java application. However, you first need
to import your project. Import steps vary depending on your IDE and build system. Most IDEs can
import Maven projects directly. For example, Eclipse users can select Import… → Existing Maven
Projects from the File menu.
If you cannot directly import your project into your IDE, you may be able to generate IDE metadata
by using a build plugin. Maven includes plugins for Eclipse and IDEA. Gradle offers plugins for
various IDEs.
If you accidentally run a web application twice, you see a “Port already in use” error.
TIP Spring Tools users can use the Relaunch button rather than the Run button to ensure
that any existing instance is closed.
If you use the Spring Boot Maven or Gradle plugins to create an executable jar, you can run your
application using java -jar, as shown in the following example:
It is also possible to run a packaged application with remote debugging support enabled. Doing so
lets you attach a debugger to your packaged application, as shown in the following example:
The Spring Boot Maven plugin includes a run goal that can be used to quickly compile and run your
application. Applications run in an exploded form, as they do in your IDE. The following example
shows a typical Maven command to run a Spring Boot application:
$ mvn spring-boot:run
40
You might also want to use the MAVEN_OPTS operating system environment variable, as shown in the
following example:
$ export MAVEN_OPTS=-Xmx1024m
The Spring Boot Gradle plugin also includes a bootRun task that can be used to run your application
in an exploded form. The bootRun task is added whenever you apply the org.springframework.boot
and java plugins and is shown in the following example:
$ gradle bootRun
You might also want to use the JAVA_OPTS operating system environment variable, as shown in the
following example:
$ export JAVA_OPTS=-Xmx1024m
Since Spring Boot applications are plain Java applications, JVM hot-swapping should work out of
the box. JVM hot swapping is somewhat limited with the bytecode that it can replace. For a more
complete solution, JRebel can be used.
The spring-boot-devtools module also includes support for quick application restarts. See the Hot
swapping “How-to” for details.
Maven
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
<optional>true</optional>
</dependency>
</dependencies>
41
Gradle
dependencies {
developmentOnly("org.springframework.boot:spring-boot-devtools")
}
Repackaged archives do not contain devtools by default. If you want to use a certain
remote devtools feature, you need to include it. When using the Maven plugin, set the
TIP
excludeDevtools property to false. When using the Gradle plugin, configure the task’s
classpath to include the developmentOnly configuration.
As described in the Restart vs Reload section, restart functionality is implemented by using two
classloaders. For most applications, this approach works well. However, it can sometimes cause
classloading issues, in particular in multi-module projects.
To diagnose whether the classloading issues are indeed caused by devtools and its two classloaders,
try disabling restart. If this solves your problems, customize the restart classloader to include your
entire project.
Several of the libraries supported by Spring Boot use caches to improve performance. For example,
template engines cache compiled templates to avoid repeatedly parsing template files. Also, Spring
MVC can add HTTP caching headers to responses when serving static resources.
42
Cache options are usually configured by settings in your application.properties file. For example,
Thymeleaf offers the spring.thymeleaf.cache property. Rather than needing to set these properties
manually, the spring-boot-devtools module automatically applies sensible development-time
configuration.
The following table lists all the properties that are applied:
If you do not want property defaults to be applied you can set spring.devtools.add-
NOTE
properties to false in your application.properties.
Because you need more information about web requests while developing Spring MVC and Spring
WebFlux applications, developer tools suggests you to enable DEBUG logging for the web logging
group. This will give you information about the incoming request, which handler is processing it,
the response outcome, and other details. If you wish to log all request details (including potentially
sensitive information), you can turn on the spring.mvc.log-request-details or spring.codec.log-
request-details configuration properties.
Applications that use spring-boot-devtools automatically restart whenever files on the classpath
change. This can be a useful feature when working in an IDE, as it gives a very fast feedback loop
for code changes. By default, any entry on the classpath that points to a directory is monitored for
changes. Note that certain resources, such as static assets and view templates, do not need to restart
the application.
43
Triggering a restart
As DevTools monitors classpath resources, the only way to trigger a restart is to update the
classpath. Whether you’re using an IDE or one of the build plugins, the modified files have to
be recompiled to trigger a restart. The way in which you cause the classpath to be updated
depends on the tool that you are using:
• In Eclipse, saving a modified file causes the classpath to be updated and triggers a restart.
• In IntelliJ IDEA, building the project (Build +→+ Build Project) has the same effect.
• If using a build plugin, running mvn compile for Maven or gradle build for Gradle will
trigger a restart.
If you are restarting with Maven or Gradle using the build plugin you must leave
NOTE the forking set to enabled. If you disable forking, the isolated application classloader
used by devtools will not be created and restarts will not operate properly.
Automatic restart works very well when used with LiveReload. See the LiveReload
section for details. If you use JRebel, automatic restarts are disabled in favor of
TIP
dynamic class reloading. Other devtools features (such as LiveReload and property
overrides) can still be used.
44
Restart vs Reload
The restart technology provided by Spring Boot works by using two classloaders. Classes that
do not change (for example, those from third-party jars) are loaded into a base classloader.
Classes that you are actively developing are loaded into a restart classloader. When the
application is restarted, the restart classloader is thrown away and a new one is created. This
approach means that application restarts are typically much faster than “cold starts”, since
the base classloader is already available and populated.
If you find that restarts are not quick enough for your applications or you encounter
classloading issues, you could consider reloading technologies such as JRebel from
ZeroTurnaround. These work by rewriting classes as they are loaded to make them more
amenable to reloading.
By default, each time your application restarts, a report showing the condition evaluation delta is
logged. The report shows the changes to your application’s auto-configuration as you make changes
such as adding or removing beans and setting configuration properties.
Properties
spring.devtools.restart.log-condition-evaluation-delta=false
Yaml
spring:
devtools:
restart:
log-condition-evaluation-delta: false
Excluding Resources
Certain resources do not necessarily need to trigger a restart when they are changed. For example,
Thymeleaf templates can be edited in-place. By default, changing resources in /META-INF/maven,
/META-INF/resources, /resources, /static, /public, or /templates does not trigger a restart but does
trigger a live reload. If you want to customize these exclusions, you can use the
spring.devtools.restart.exclude property. For example, to exclude only /static and /public you
would set the following property:
Properties
spring.devtools.restart.exclude=static/**,public/**
45
Yaml
spring:
devtools:
restart:
exclude: "static/**,public/**"
If you want to keep those defaults and add additional exclusions, use the
TIP
spring.devtools.restart.additional-exclude property instead.
You may want your application to be restarted or reloaded when you make changes to files that are
not on the classpath. To do so, use the spring.devtools.restart.additional-paths property to
configure additional paths to watch for changes. You can use the spring.devtools.restart.exclude
property described earlier to control whether changes beneath the additional paths trigger a full
restart or a live reload.
Disabling Restart
If you do not want to use the restart feature, you can disable it by using the
spring.devtools.restart.enabled property. In most cases, you can set this property in your
application.properties (doing so still initializes the restart classloader, but it does not watch for file
changes).
If you need to completely disable restart support (for example, because it does not work with a
specific library), you need to set the spring.devtools.restart.enabled System property to false
before calling SpringApplication.run(…), as shown in the following example:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
46
Kotlin
import org.springframework.boot.SpringApplication
import org.springframework.boot.autoconfigure.SpringBootApplication
@SpringBootApplication
object MyApplication {
@JvmStatic
fun main(args: Array<String>) {
System.setProperty("spring.devtools.restart.enabled", "false")
SpringApplication.run(MyApplication::class.java, *args)
}
If you work with an IDE that continuously compiles changed files, you might prefer to trigger
restarts only at specific times. To do so, you can use a “trigger file”, which is a special file that must
be modified when you want to actually trigger a restart check.
Any update to the file will trigger a check, but restart only actually occurs if
NOTE
Devtools has detected it has something to do.
To use a trigger file, set the spring.devtools.restart.trigger-file property to the name (excluding
any path) of your trigger file. The trigger file must appear somewhere on your classpath.
src
+- main
+- resources
+- .reloadtrigger
Properties
spring.devtools.restart.trigger-file=.reloadtrigger
Yaml
spring:
devtools:
restart:
trigger-file: ".reloadtrigger"
47
Restarts will now only happen when the src/main/resources/.reloadtrigger is updated.
Some IDEs have features that save you from needing to update your trigger file manually. Spring
Tools for Eclipse and IntelliJ IDEA (Ultimate Edition) both have such support. With Spring Tools, you
can use the “reload” button from the console view (as long as your trigger-file is named
.reloadtrigger). For IntelliJ IDEA, you can follow the instructions in their documentation.
As described earlier in the Restart vs Reload section, restart functionality is implemented by using
two classloaders. If this causes issues, you might need to customize what gets loaded by which
classloader.
By default, any open project in your IDE is loaded with the “restart” classloader, and any regular
.jar file is loaded with the “base” classloader. The same is true if you use mvn spring-boot:run or
gradle bootRun: the project containing your @SpringBootApplication is loaded with the “restart”
classloader, and everything else with the “base” classloader.
You can instruct Spring Boot to load parts of your project with a different classloader by creating a
META-INF/spring-devtools.properties file. The spring-devtools.properties file can contain
properties prefixed with restart.exclude and restart.include. The include elements are items that
should be pulled up into the “restart” classloader, and the exclude elements are items that should be
pushed down into the “base” classloader. The value of the property is a regex pattern that is applied
to the classpath, as shown in the following example:
Properties
restart.exclude.companycommonlibs=/mycorp-common-[\\w\\d-\\.]+\\.jar
restart.include.projectcommon=/mycorp-myproj-[\\w\\d-\\.]+\\.jar
Yaml
restart:
exclude:
companycommonlibs: "/mycorp-common-[\\w\\d-\\.]+\\.jar"
include:
projectcommon: "/mycorp-myproj-[\\w\\d-\\.]+\\.jar"
All property keys must be unique. As long as a property starts with restart.include.
NOTE
or restart.exclude. it is considered.
48
Known Limitations
Restart functionality does not work well with objects that are deserialized by using a standard
ObjectInputStream. If you need to deserialize data, you may need to use Spring’s
ConfigurableObjectInputStream in combination with
Thread.currentThread().getContextClassLoader().
Unfortunately, several third-party libraries deserialize without considering the context classloader.
If you find such a problem, you need to request a fix with the original authors.
6.8.4. LiveReload
The spring-boot-devtools module includes an embedded LiveReload server that can be used to
trigger a browser refresh when a resource is changed. LiveReload browser extensions are freely
available for Chrome, Firefox and Safari from livereload.com.
If you do not want to start the LiveReload server when your application runs, you can set the
spring.devtools.livereload.enabled property to false.
You can only run one LiveReload server at a time. Before starting your application,
NOTE ensure that no other LiveReload servers are running. If you start multiple
applications from your IDE, only the first has LiveReload support.
WARNING To trigger LiveReload when a file changes, Automatic Restart must be enabled.
You can configure global devtools settings by adding any of the following files to the
$HOME/.config/spring-boot directory:
1. spring-boot-devtools.properties
2. spring-boot-devtools.yaml
3. spring-boot-devtools.yml
Any properties added to these files apply to all Spring Boot applications on your machine that use
devtools. For example, to configure restart to always use a trigger file, you would add the following
property to your spring-boot-devtools file:
Properties
spring.devtools.restart.trigger-file=.reloadtrigger
49
Yaml
spring:
devtools:
restart:
trigger-file: ".reloadtrigger"
By default, $HOME is the user’s home directory. To customize this location, set the
SPRING_DEVTOOLS_HOME environment variable or the spring.devtools.home system property.
FileSystemWatcher works by polling the class changes with a certain time interval, and then
waiting for a predefined quiet period to make sure there are no more changes. Since Spring Boot
relies entirely on the IDE to compile and copy files into the location from where Spring Boot can
read them, you might find that there are times when certain changes are not reflected when
devtools restarts the application. If you observe such problems constantly, try increasing the
spring.devtools.restart.poll-interval and spring.devtools.restart.quiet-period parameters to
the values that fit your development environment:
Properties
spring.devtools.restart.poll-interval=2s
spring.devtools.restart.quiet-period=1s
Yaml
spring:
devtools:
restart:
poll-interval: "2s"
quiet-period: "1s"
The monitored classpath directories are now polled every 2 seconds for changes, and a 1 second
quiet period is maintained to make sure there are no additional class changes.
50
6.8.6. Remote Applications
The Spring Boot developer tools are not limited to local development. You can also use several
features when running applications remotely. Remote support is opt-in as enabling it can be a
security risk. It should only be enabled when running on a trusted network or when secured with
SSL. If neither of these options is available to you, you should not use DevTools' remote support.
You should never enable support on a production deployment.
To enable it, you need to make sure that devtools is included in the repackaged archive, as shown in
the following listing:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<excludeDevtools>false</excludeDevtools>
</configuration>
</plugin>
</plugins>
</build>
Then you need to set the spring.devtools.remote.secret property. Like any important password or
secret, the value should be unique and strong such that it cannot be guessed or brute-forced.
Remote devtools support is provided in two parts: a server-side endpoint that accepts connections
and a client application that you run in your IDE. The server component is automatically enabled
when the spring.devtools.remote.secret property is set. The client component must be launched
manually.
The remote client application is designed to be run from within your IDE. You need to run
org.springframework.boot.devtools.RemoteSpringApplication with the same classpath as the remote
project that you connect to. The application’s single required argument is the remote URL to which
it connects.
For example, if you are using Eclipse or Spring Tools and you have a project named my-app that you
have deployed to Cloud Foundry, you would do the following:
51
• Add https://myapp.cfapps.io to the Program arguments (or whatever your remote URL is).
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ ___ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | | _ \___ _ __ ___| |_ ___ \ \ \ \
\\/ ___)| |_)| | | | | || (_| []::::::[] / -_) ' \/ _ \ _/ -_) ) ) ) )
' |____| .__|_| |_|_| |_\__, | |_|_\___|_|_|_\___/\__\___|/ / / /
=========|_|==============|___/===================================/_/_/_/
:: Spring Boot Remote :: (v3.1.0-RC2)
Because the remote client is using the same classpath as the real application it can
NOTE directly read application properties. This is how the spring.devtools.remote.secret
property is read and passed to the server for authentication.
If you need to use a proxy to access the remote application, configure the
TIP
spring.devtools.remote.proxy.host and spring.devtools.remote.proxy.port properties.
Remote Update
The remote client monitors your application classpath for changes in the same way as the local
restart. Any updated resource is pushed to the remote application and (if required) triggers a
restart. This can be helpful if you iterate on a feature that uses a cloud service that you do not have
locally. Generally, remote updates and restarts are much quicker than a full rebuild and deploy
cycle.
On a slower development environment, it may happen that the quiet period is not enough, and the
changes in the classes may be split into batches. The server is restarted after the first batch of class
changes is uploaded. The next batch can’t be sent to the application, since the server is restarting.
52
This is typically manifested by a warning in the RemoteSpringApplication logs about failing to upload
some of the classes, and a consequent retry. But it may also lead to application code inconsistency
and failure to restart after the first batch of changes is uploaded. If you observe such problems
constantly, try increasing the spring.devtools.restart.poll-interval and
spring.devtools.restart.quiet-period parameters to the values that fit your development
environment. See the Configuring File System Watcher section for configuring these properties.
Files are only monitored when the remote client is running. If you change a file
NOTE
before starting the remote client, it is not pushed to the remote server.
For additional “production ready” features, such as health, auditing, and metric REST or JMX end-
points, consider adding spring-boot-actuator. See Production-ready Features for details.
53
Chapter 7. Core Features
This section dives into the details of Spring Boot. Here you can learn about the key features that you
may want to use and customize. If you have not already done so, you might want to read the
"Getting Started" and "Developing with Spring Boot" sections, so that you have a good grounding of
the basics.
7.1. SpringApplication
The SpringApplication class provides a convenient way to bootstrap a Spring application that is
started from a main() method. In many situations, you can delegate to the static
SpringApplication.run method, as shown in the following example:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
When your application starts, you should see something similar to the following output:
54
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
By default, INFO logging messages are shown, including some relevant startup details, such as the
user that launched the application. If you need a log level other than INFO, you can set it, as
described in Log Levels. The application version is determined using the implementation version
from the main application class’s package. Startup information logging can be turned off by setting
spring.main.log-startup-info to false. This will also turn off logging of the application’s active
profiles.
If your application fails to start, registered FailureAnalyzers get a chance to provide a dedicated
error message and a concrete action to fix the problem. For instance, if you start a web application
on port 8080 and that port is already in use, you should see something similar to the following
55
message:
***************************
APPLICATION FAILED TO START
***************************
Description:
Embedded servlet container failed to start. Port 8080 was already in use.
Action:
Identify and stop the process that is listening on port 8080 or configure this
application to listen on another port.
Spring Boot provides numerous FailureAnalyzer implementations, and you can add
NOTE
your own.
If no failure analyzers are able to handle the exception, you can still display the full conditions
report to better understand what went wrong. To do so, you need to enable the debug property or
enable DEBUG logging for
org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener.
For instance, if you are running your application by using java -jar, you can enable the debug
property as follows:
A downside of lazy initialization is that it can delay the discovery of a problem with the application.
If a misconfigured bean is initialized lazily, a failure will no longer occur during startup and the
problem will only become apparent when the bean is initialized. Care must also be taken to ensure
that the JVM has sufficient memory to accommodate all of the application’s beans and not just those
that are initialized during startup. For these reasons, lazy initialization is not enabled by default
and it is recommended that fine-tuning of the JVM’s heap size is done before enabling lazy
initialization.
56
example:
Properties
spring.main.lazy-initialization=true
Yaml
spring:
main:
lazy-initialization: true
If you want to disable lazy initialization for certain beans while using lazy
TIP initialization for the rest of the application, you can explicitly set their lazy attribute to
false using the @Lazy(false) annotation.
The banner that is printed on start up can be changed by adding a banner.txt file to your classpath
or by setting the spring.banner.location property to the location of such a file. If the file has an
encoding other than UTF-8, you can set spring.banner.charset.
Inside your banner.txt file, you can use any key available in the Environment as well as any of the
following placeholders:
Variable Description
${application.version} The version number of your application, as
declared in MANIFEST.MF. For example,
Implementation-Version: 1.0 is printed as 1.0.
${application.formatted-version} The version number of your application, as
declared in MANIFEST.MF and formatted for
display (surrounded with brackets and prefixed
with v). For example (v1.0).
${spring-boot.version} The Spring Boot version that you are using. For
example 3.1.0-RC2.
${spring-boot.formatted-version} The Spring Boot version that you are using,
formatted for display (surrounded with brackets
and prefixed with v). For example (v3.1.0-RC2).
${Ansi.NAME} (or ${AnsiColor.NAME}, Where NAME is the name of an ANSI escape code.
${AnsiBackground.NAME}, ${AnsiStyle.NAME}) See AnsiPropertySource for details.
${application.title} The title of your application, as declared in
MANIFEST.MF. For example Implementation-Title:
MyApp is printed as MyApp.
57
The SpringApplication.setBanner(…) method can be used if you want to generate a
TIP banner programmatically. Use the org.springframework.boot.Banner interface and
implement your own printBanner() method.
You can also use the spring.main.banner-mode property to determine if the banner has to be printed
on System.out (console), sent to the configured logger (log), or not produced at all (off).
The printed banner is registered as a singleton bean under the following name: springBootBanner.
This is why we recommend that you always launch unpacked jars using java
org.springframework.boot.loader.JarLauncher. This will initialize the application.*
banner variables before building the classpath and launching your app.
If the SpringApplication defaults are not to your taste, you can instead create a local instance and
customize it. For example, to turn off the banner, you could write:
Java
import org.springframework.boot.Banner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
58
Kotlin
import org.springframework.boot.Banner
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
For a complete list of the configuration options, see the SpringApplication Javadoc.
The SpringApplicationBuilder lets you chain together multiple method calls and includes parent and
child methods that let you create a hierarchy, as shown in the following example:
Java
new SpringApplicationBuilder().sources(Parent.class)
.child(Application.class)
.bannerMode(Banner.Mode.OFF)
.run(args);
Kotlin
SpringApplicationBuilder()
.sources(Parent::class.java)
.child(Application::class.java)
.bannerMode(Banner.Mode.OFF)
.run(*args)
59
There are some restrictions when creating an ApplicationContext hierarchy. For
example, Web components must be contained within the child context, and the
NOTE
same Environment is used for both parent and child contexts. See the
SpringApplicationBuilder Javadoc for full details.
When deployed on platforms, applications can provide information about their availability to the
platform using infrastructure such as Kubernetes Probes. Spring Boot includes out-of-the box
support for the commonly used “liveness” and “readiness” availability states. If you are using
Spring Boot’s “actuator” support then these states are exposed as health endpoint groups.
In addition, you can also obtain availability states by injecting the ApplicationAvailability interface
into your own beans.
Liveness State
The “Liveness” state of an application tells whether its internal state allows it to work correctly, or
recover by itself if it is currently failing. A broken “Liveness” state means that the application is in a
state that it cannot recover from, and the infrastructure should restart the application.
In general, the "Liveness" state should not be based on external checks, such as
NOTE Health checks. If it did, a failing external system (a database, a Web API, an external
cache) would trigger massive restarts and cascading failures across the platform.
The internal state of Spring Boot applications is mostly represented by the Spring
ApplicationContext. If the application context has started successfully, Spring Boot assumes that the
application is in a valid state. An application is considered live as soon as the context has been
refreshed, see Spring Boot application lifecycle and related Application Events.
Readiness State
The “Readiness” state of an application tells whether the application is ready to handle traffic. A
failing “Readiness” state tells the platform that it should not route traffic to the application for now.
This typically happens during startup, while CommandLineRunner and ApplicationRunner components
are being processed, or at any time if the application decides that it is too busy for additional traffic.
An application is considered ready as soon as application and command-line runners have been
called, see Spring Boot application lifecycle and related Application Events.
Application components can retrieve the current availability state at any time, by injecting the
ApplicationAvailability interface and calling methods on it. More often, applications will want to
60
listen to state updates or update the state of the application.
For example, we can export the "Readiness" state of the application to a file so that a Kubernetes
"exec Probe" can look at this file:
Java
import org.springframework.boot.availability.AvailabilityChangeEvent;
import org.springframework.boot.availability.ReadinessState;
import org.springframework.context.event.EventListener;
import org.springframework.stereotype.Component;
@Component
public class MyReadinessStateExporter {
@EventListener
public void onStateChange(AvailabilityChangeEvent<ReadinessState> event) {
switch (event.getState()) {
case ACCEPTING_TRAFFIC:
// create file /tmp/healthy
break;
case REFUSING_TRAFFIC:
// remove file /tmp/healthy
break;
}
}
61
Kotlin
import org.springframework.boot.availability.AvailabilityChangeEvent
import org.springframework.boot.availability.ReadinessState
import org.springframework.context.event.EventListener
import org.springframework.stereotype.Component
@Component
class MyReadinessStateExporter {
@EventListener
fun onStateChange(event: AvailabilityChangeEvent<ReadinessState?>) {
when (event.state) {
ReadinessState.ACCEPTING_TRAFFIC -> {
// create file /tmp/healthy
}
ReadinessState.REFUSING_TRAFFIC -> {
// remove file /tmp/healthy
}
else -> {
// ...
}
}
}
We can also update the state of the application, when the application breaks and cannot recover:
62
Java
import org.springframework.boot.availability.AvailabilityChangeEvent;
import org.springframework.boot.availability.LivenessState;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Component;
@Component
public class MyLocalCacheVerifier {
Kotlin
import org.springframework.boot.availability.AvailabilityChangeEvent
import org.springframework.boot.availability.LivenessState
import org.springframework.context.ApplicationEventPublisher
import org.springframework.stereotype.Component
@Component
class MyLocalCacheVerifier(private val eventPublisher: ApplicationEventPublisher) {
fun checkLocalCache() {
try {
// ...
} catch (ex: CacheCompletelyBrokenException) {
AvailabilityChangeEvent.publish(eventPublisher, ex, LivenessState.BROKEN)
}
}
Spring Boot provides Kubernetes HTTP probes for "Liveness" and "Readiness" with Actuator Health
Endpoints. You can get more guidance about deploying Spring Boot applications on Kubernetes in
63
the dedicated section.
Some events are actually triggered before the ApplicationContext is created, so you
cannot register a listener on those as a @Bean. You can register them with the
SpringApplication.addListeners(…) method or the
SpringApplicationBuilder.listeners(…) method.
If you want those listeners to be registered automatically, regardless of the way the
application is created, you can add a META-INF/spring.factories file to your project
NOTE
and reference your listener(s) by using the
org.springframework.context.ApplicationListener key, as shown in the following
example:
org.springframework.context.ApplicationListener=com.example.project.MyLi
stener
Application events are sent in the following order, as your application runs:
1. An ApplicationStartingEvent is sent at the start of a run but before any processing, except for
the registration of listeners and initializers.
4. An ApplicationPreparedEvent is sent just before the refresh is started but after bean definitions
have been loaded.
5. An ApplicationStartedEvent is sent after the context has been refreshed but before any
application and command-line runners have been called.
7. An ApplicationReadyEvent is sent after any application and command-line runners have been
called.
The above list only includes SpringApplicationEvents that are tied to a SpringApplication. In
addition to these, the following events are also published after ApplicationPreparedEvent and before
ApplicationStartedEvent:
64
• A WebServerInitializedEvent is sent after the WebServer is ready.
ServletWebServerInitializedEvent and ReactiveWebServerInitializedEvent are the servlet and
reactive variants respectively.
You often need not use application events, but it can be handy to know that they exist.
TIP
Internally, Spring Boot uses events to handle a variety of tasks.
Event listeners should not run potentially lengthy tasks as they execute in the same
NOTE
thread by default. Consider using application and command-line runners instead.
Application events are sent by using Spring Framework’s event publishing mechanism. Part of this
mechanism ensures that an event published to the listeners in a child context is also published to
the listeners in any ancestor contexts. As a result of this, if your application uses a hierarchy of
SpringApplication instances, a listener may receive multiple instances of the same type of
application event.
To allow your listener to distinguish between an event for its context and an event for a descendant
context, it should request that its application context is injected and then compare the injected
context with the context of the event. The context can be injected by implementing
ApplicationContextAware or, if the listener is a bean, by using @Autowired.
A SpringApplication attempts to create the right type of ApplicationContext on your behalf. The
algorithm used to determine a WebApplicationType is the following:
This means that if you are using Spring MVC and the new WebClient from Spring WebFlux in the
same application, Spring MVC will be used by default. You can override that easily by calling
setWebApplicationType(WebApplicationType).
It is also possible to take complete control of the ApplicationContext type that is used by calling
setApplicationContextFactory(…).
If you need to access the application arguments that were passed to SpringApplication.run(…), you
can inject a org.springframework.boot.ApplicationArguments bean. The ApplicationArguments
interface provides access to both the raw String[] arguments as well as parsed option and non-
65
option arguments, as shown in the following example:
Java
import java.util.List;
import org.springframework.boot.ApplicationArguments;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.boot.ApplicationArguments
import org.springframework.stereotype.Component
@Component
class MyBean(args: ApplicationArguments) {
init {
val debug = args.containsOption("debug")
val files = args.nonOptionArgs
if (debug) {
println(files)
}
// if run with "--debug logfile.txt" prints ["logfile.txt"]
}
If you need to run some specific code once the SpringApplication has started, you can implement
the ApplicationRunner or CommandLineRunner interfaces. Both interfaces work in the same way and
66
offer a single run method, which is called just before SpringApplication.run(…) completes.
This contract is well suited for tasks that should run after application startup but
NOTE
before it starts accepting traffic.
Java
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;
@Component
public class MyCommandLineRunner implements CommandLineRunner {
@Override
public void run(String... args) {
// Do something...
}
Kotlin
import org.springframework.boot.CommandLineRunner
import org.springframework.stereotype.Component
@Component
class MyCommandLineRunner : CommandLineRunner {
If several CommandLineRunner or ApplicationRunner beans are defined that must be called in a specific
order, you can additionally implement the org.springframework.core.Ordered interface or use the
org.springframework.core.annotation.Order annotation.
Each SpringApplication registers a shutdown hook with the JVM to ensure that the
ApplicationContext closes gracefully on exit. All the standard Spring lifecycle callbacks (such as the
DisposableBean interface or the @PreDestroy annotation) can be used.
67
wish to return a specific exit code when SpringApplication.exit() is called. This exit code can then
be passed to System.exit() to return it as a status code, as shown in the following example:
Java
import org.springframework.boot.ExitCodeGenerator;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
@SpringBootApplication
public class MyApplication {
@Bean
public ExitCodeGenerator exitCodeGenerator() {
return () -> 42;
}
Kotlin
import org.springframework.boot.ExitCodeGenerator
import org.springframework.boot.SpringApplication
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
import org.springframework.context.annotation.Bean
import kotlin.system.exitProcess
@SpringBootApplication
class MyApplication {
@Bean
fun exitCodeGenerator() = ExitCodeGenerator { 42 }
Also, the ExitCodeGenerator interface may be implemented by exceptions. When such an exception
is encountered, Spring Boot returns the exit code provided by the implemented getExitCode()
68
method.
If there is more than one ExitCodeGenerator, the first non-zero exit code that is generated is used. To
control the order in which the generators are called, additionally implement the
org.springframework.core.Ordered interface or use the org.springframework.core.annotation.Order
annotation.
If you want to know on which HTTP port the application is running, get the property
TIP
with a key of local.server.port.
During the application startup, the SpringApplication and the ApplicationContext perform many
tasks related to the application lifecycle, the beans lifecycle or even processing application events.
With ApplicationStartup, Spring Framework allows you to track the application startup sequence
with StartupStep objects. This data can be collected for profiling purposes, or just to have a better
understanding of an application startup process.
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.context.metrics.buffering.BufferingApplicationStartup;
@SpringBootApplication
public class MyApplication {
69
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.context.metrics.buffering.BufferingApplicationStartup
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
Spring Boot ships with the BufferingApplicationStartup variant; this implementation is meant for
buffering the startup steps and draining them into an external metrics system. Applications can ask
for the bean of type BufferingApplicationStartup in any component.
Spring Boot can also be configured to expose a startup endpoint that provides this information as a
JSON document.
Property values can be injected directly into your beans by using the @Value annotation, accessed
through Spring’s Environment abstraction, or be bound to structured objects through
@ConfigurationProperties.
Spring Boot uses a very particular PropertySource order that is designed to allow sensible
overriding of values. Later property sources can override the values defined in earlier ones.
Sources are considered in the following order:
2. @PropertySource annotations on your @Configuration classes. Please note that such property
sources are not added to the Environment until the application context is being refreshed. This is
70
too late to configure certain properties such as logging.* and spring.main.* which are read
before refresh begins.
5. OS environment variables.
12. properties attribute on your tests. Available on @SpringBootTest and the test annotations for
testing a particular slice of your application.
14. Devtools global settings properties in the $HOME/.config/spring-boot directory when devtools is
active.
1. Application properties packaged inside your jar (application.properties and YAML variants).
It is recommended to stick with one format for your entire application. If you have
NOTE configuration files with both .properties and YAML format in the same location,
.properties takes precedence.
To provide a concrete example, suppose you develop a @Component that uses a name property, as
shown in the following example:
71
Java
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
@Value("${name}")
private String name;
// ...
Kotlin
import org.springframework.beans.factory.annotation.Value
import org.springframework.stereotype.Component
@Component
class MyBean {
@Value("\${name}")
private val name: String? = null
// ...
On your application classpath (for example, inside your jar) you can have an
application.properties file that provides a sensible default property value for name. When running
in a new environment, an application.properties file can be provided outside of your jar that
overrides the name. For one-off testing, you can launch with a specific command line switch (for
example, java -jar app.jar --name="Spring").
The env and configprops endpoints can be useful in determining why a property has a
TIP particular value. You can use these two endpoints to diagnose unexpected property
values. See the "Production ready features" section for details.
By default, SpringApplication converts any command line option arguments (that is, arguments
starting with --, such as --server.port=9000) to a property and adds them to the Spring Environment.
As mentioned previously, command line properties always take precedence over file-based
property sources.
If you do not want command line properties to be added to the Environment, you can disable them
by using SpringApplication.setAddCommandLineProperties(false).
72
7.2.2. JSON Application Properties
Environment variables and system properties often have restrictions that mean some property
names cannot be used. To help with this, Spring Boot allows you to encode a block of properties into
a single JSON structure.
For example, the SPRING_APPLICATION_JSON property can be supplied on the command line in a UN*X
shell as an environment variable:
In the preceding example, you end up with my.name=test in the Spring Environment.
If you are deploying to a classic Application Server, you could also use a JNDI variable named
java:comp/env/spring.application.json.
Although null values from the JSON will be added to the resulting property source,
the PropertySourcesPropertyResolver treats null properties as missing values. This
NOTE
means that the JSON cannot override properties from lower order property sources
with a null value.
Spring Boot will automatically find and load application.properties and application.yaml files from
the following locations when your application starts:
73
The list is ordered by precedence (with values from lower items overriding earlier ones).
Documents from the loaded files are added as PropertySources to the Spring Environment.
If you do not like application as the configuration file name, you can switch to another file name by
specifying a spring.config.name environment property. For example, to look for
myproject.properties and myproject.yaml files you can run your application as follows:
You can also refer to an explicit location by using the spring.config.location environment property.
This property accepts a comma-separated list of one or more locations to check.
Use the prefix optional: if the locations are optional and you do not mind if they do
TIP
not exist.
If spring.config.location contains directories (as opposed to files), they should end in /. At runtime
they will be appended with the names generated from spring.config.name before being loaded. Files
specified in spring.config.location are imported directly.
Both directory and file location values are also expanded to check for profile-
specific files. For example, if you have a spring.config.location of
NOTE
classpath:myconfig.properties, you will also find appropriate classpath:myconfig-
<profile>.properties files are loaded.
In most situations, each spring.config.location item you add will reference a single file or
directory. Locations are processed in the order that they are defined and later ones can override
the values of earlier ones.
If you have a complex location setup, and you use profile-specific configuration files, you may need
to provide further hints so that Spring Boot knows how they should be grouped. A location group is
a collection of locations that are all considered at the same level. For example, you might want to
group all classpath locations, then all external locations. Items within a location group should be
separated with ;. See the example in the “Profile Specific Files” section for more details.
Locations configured by using spring.config.location replace the default locations. For example, if
74
spring.config.location is configured with the value optional:classpath:/custom-
config/,optional:file:./custom-config/, the complete set of locations considered is:
1. optional:classpath:custom-config/
2. optional:file:./custom-config/
If you prefer to add additional locations, rather than replacing them, you can use
spring.config.additional-location. Properties loaded from additional locations can override those
in the default locations. For example, if spring.config.additional-location is configured with the
value optional:classpath:/custom-config/,optional:file:./custom-config/, the complete set of
locations considered is:
1. optional:classpath:/;optional:classpath:/config/
2. optional:file:./;optional:file:./config/;optional:file:./config/*/
3. optional:classpath:custom-config/
4. optional:file:./custom-config/
This search ordering lets you specify default values in one configuration file and then selectively
override those values in another. You can provide default values for your application in
application.properties (or whatever other basename you choose with spring.config.name) in one of
the default locations. These default values can then be overridden at runtime with a different file
located in one of the custom locations.
If you use environment variables rather than system properties, most operating
systems disallow period-separated key names, but you can use underscores instead
NOTE
(for example, SPRING_CONFIG_NAME instead of spring.config.name). See Binding From
Environment Variables for details.
Optional Locations
By default, when a specified config data location does not exist, Spring Boot will throw a
ConfigDataLocationNotFoundException and your application will not start.
If you want to specify a location, but you do not mind if it does not always exist, you can use the
optional: prefix. You can use this prefix with the spring.config.location and
spring.config.additional-location properties, as well as with spring.config.import declarations.
If you want to ignore all ConfigDataLocationNotFoundExceptions and always continue to start your
application, you can use the spring.config.on-not-found property. Set the value to ignore using
SpringApplication.setDefaultProperties(…) or with a system/environment variable.
75
Wildcard Locations
If a config file location includes the * character for the last path segment, it is considered a wildcard
location. Wildcards are expanded when the config is loaded so that immediate subdirectories are
also checked. Wildcard locations are particularly useful in an environment such as Kubernetes
when there are multiple sources of config properties.
For example, if you have some Redis configuration and some MySQL configuration, you might want
to keep those two pieces of configuration separate, while requiring that both those are present in
an application.properties file. This might result in two separate application.properties files
mounted at different locations such as /config/redis/application.properties and
/config/mysql/application.properties. In such a case, having a wildcard location of config/*/, will
result in both files being processed.
By default, Spring Boot includes config/*/ in the default search locations. It means that all
subdirectories of the /config directory outside of your jar will be searched.
You can use wildcard locations yourself with the spring.config.location and
spring.config.additional-location properties.
A wildcard location must contain only one * and end with */ for search locations
NOTE that are directories or */<filename> for search locations that are files. Locations with
wildcards are sorted alphabetically based on the absolute path of the file names.
Wildcard locations only work with external directories. You cannot use a wildcard in a
TIP
classpath: location.
As well as application property files, Spring Boot will also attempt to load profile-specific files using
the naming convention application-{profile}. For example, if your application activates a profile
named prod and uses YAML files, then both application.yaml and application-prod.yaml will be
considered.
Profile-specific properties are loaded from the same locations as standard application.properties,
with profile-specific files always overriding the non-specific ones. If several profiles are specified, a
last-wins strategy applies. For example, if profiles prod,live are specified by the
spring.profiles.active property, values in application-prod.properties can be overridden by those
in application-live.properties.
76
The last-wins strategy applies at the location group level. A spring.config.location
of classpath:/cfg/,classpath:/ext/ will not have the same override rules as
classpath:/cfg/;classpath:/ext/.
For example, continuing our prod,live example above, we might have the following
files:
/cfg
application-live.properties
/ext
application-live.properties
application-prod.properties
1. /cfg/application-live.properties
2. /ext/application-prod.properties
3. /ext/application-live.properties
1. /ext/application-prod.properties
2. /cfg/application-live.properties
3. /ext/application-live.properties
The Environment has a set of default profiles (by default, [default]) that are used if no active profiles
are set. In other words, if no profiles are explicitly activated, then properties from application-
default are considered.
Properties files are only ever loaded once. If you have already directly imported a
NOTE
profile specific property files then it will not be imported a second time.
Application properties may import further config data from other locations using the
spring.config.import property. Imports are processed as they are discovered, and are treated as
additional documents inserted immediately below the one that declares the import.
For example, you might have the following in your classpath application.properties file:
Properties
spring.application.name=myapp
spring.config.import=optional:file:./dev.properties
77
Yaml
spring:
application:
name: "myapp"
config:
import: "optional:file:./dev.properties"
This will trigger the import of a dev.properties file in current directory (if such a file exists). Values
from the imported dev.properties will take precedence over the file that triggered the import. In
the above example, the dev.properties could redefine spring.application.name to a different value.
An import will only be imported once no matter how many times it is declared. The order an
import is defined inside a single document within the properties/yaml file does not matter. For
instance, the two examples below produce the same result:
Properties
spring.config.import=my.properties
my.property=value
Yaml
spring:
config:
import: "my.properties"
my:
property: "value"
Properties
my.property=value
spring.config.import=my.properties
Yaml
my:
property: "value"
spring:
config:
import: "my.properties"
In both of the above examples, the values from the my.properties file will take precedence over the
file that triggered its import.
Several locations can be specified under a single spring.config.import key. Locations will be
processed in the order that they are defined, with later imports taking precedence.
78
When appropriate, Profile-specific variants are also considered for import. The
NOTE example above would import both my.properties as well as any my-
<profile>.properties variants.
Spring Boot includes pluggable API that allows various different location addresses to
be supported. By default you can import Java Properties, YAML and “configuration
trees”.
Third-party jars can offer support for additional technologies (there is no requirement
TIP
for files to be local). For example, you can imagine config data being from external
stores such as Consul, Apache ZooKeeper or Netflix Archaius.
If you want to support your own locations, see the ConfigDataLocationResolver and
ConfigDataLoader classes in the org.springframework.boot.context.config package.
Some cloud platforms cannot add a file extension to volume mounted files. To import these
extensionless files, you need to give Spring Boot a hint so that it knows how to load them. You can
do this by putting an extension hint in square brackets.
For example, suppose you have a /etc/config/myconfig file that you wish to import as yaml. You can
import it from your application.properties using the following:
Properties
spring.config.import=file:/etc/config/myconfig[.yaml]
Yaml
spring:
config:
import: "file:/etc/config/myconfig[.yaml]"
When running applications on a cloud platform (such as Kubernetes) you often need to read config
values that the platform supplies. It is not uncommon to use environment variables for such
purposes, but this can have drawbacks, especially if the value is supposed to be kept secret.
As an alternative to environment variables, many cloud platforms now allow you to map
configuration into mounted data volumes. For example, Kubernetes can volume mount both
ConfigMaps and Secrets.
There are two common volume mount patterns that can be used:
2. Multiple files are written to a directory tree, with the filename becoming the ‘key’ and the
79
contents becoming the ‘value’.
For the first case, you can import the YAML or Properties file directly using spring.config.import as
described above. For the second case, you need to use the configtree: prefix so that Spring Boot
knows it needs to expose all the files as properties.
As an example, let’s imagine that Kubernetes has mounted the following volume:
etc/
config/
myapp/
username
password
The contents of the username file would be a config value, and the contents of password would be a
secret.
To import these properties, you can add the following to your application.properties or
application.yaml file:
Properties
spring.config.import=optional:configtree:/etc/config/
Yaml
spring:
config:
import: "optional:configtree:/etc/config/"
You can then access or inject myapp.username and myapp.password properties from the Environment in
the usual way.
The folders under the config tree form the property name. In the above example, to
TIP access the properties as username and password, you can set spring.config.import to
optional:configtree:/etc/config/myapp.
Filenames with dot notation are also correctly mapped. For example, in the above
NOTE example, a file named myapp.username in /etc/config would result in a
myapp.username property in the Environment.
Configuration tree values can be bound to both string String and byte[] types
TIP
depending on the contents expected.
If you have multiple config trees to import from the same parent folder you can use a wildcard
shortcut. Any configtree: location that ends with /*/ will import all immediate children as config
trees.
80
For example, given the following volume:
etc/
config/
dbconfig/
db/
username
password
mqconfig/
mq/
username
password
Properties
spring.config.import=optional:configtree:/etc/config/*/
Yaml
spring:
config:
import: "optional:configtree:/etc/config/*/"
Directories loaded using a wildcard are sorted alphabetically. If you need a different
NOTE
order, then you should list each location as a separate import
Configuration trees can also be used for Docker secrets. When a Docker swarm service is granted
access to a secret, the secret gets mounted into the container. For example, if a secret named
db.password is mounted at location /run/secrets/, you can make db.password available to the Spring
environment using the following:
Properties
spring.config.import=optional:configtree:/run/secrets/
Yaml
spring:
config:
import: "optional:configtree:/run/secrets/"
81
Property Placeholders
The values in application.properties and application.yaml are filtered through the existing
Environment when they are used, so you can refer back to previously defined values (for example,
from System properties or environment variables). The standard ${name} property-placeholder
syntax can be used anywhere within a value. Property placeholders can also specify a default value
using a : to separate the default value from the property name, for example ${name:default}.
The use of placeholders with and without defaults is shown in the following example:
Properties
app.name=MyApp
app.description=${app.name} is a Spring Boot application written by
${username:Unknown}
Yaml
app:
name: "MyApp"
description: "${app.name} is a Spring Boot application written by
${username:Unknown}"
Assuming that the username property has not been set elsewhere, app.description will have the
value MyApp is a Spring Boot application written by Unknown.
You should always refer to property names in the placeholder using their canonical
form (kebab-case using only lowercase letters). This will allow Spring Boot to use
the same logic as it does when relaxed binding @ConfigurationProperties.
NOTE
For example, ${demo.item-price} will pick up demo.item-price and demo.itemPrice
forms from the application.properties file, as well as DEMO_ITEMPRICE from the
system environment. If you used ${demo.itemPrice} instead, demo.item-price and
DEMO_ITEMPRICE would not be considered.
You can also use this technique to create “short” variants of existing Spring Boot
TIP
properties. See the Use ‘Short’ Command Line Arguments how-to for details.
Spring Boot allows you to split a single physical file into multiple logical documents which are each
added independently. Documents are processed in order, from top to bottom. Later documents can
override the properties defined in earlier ones.
For application.yaml files, the standard YAML multi-document syntax is used. Three consecutive
hyphens represent the end of one document, and the start of the next.
82
spring:
application:
name: "MyApp"
---
spring:
application:
name: "MyCloudApp"
config:
activate:
on-cloud-platform: "kubernetes"
For application.properties files a special #--- or !--- comment is used to mark the document splits:
spring.application.name=MyApp
#---
spring.application.name=MyCloudApp
spring.config.activate.on-cloud-platform=kubernetes
Property file separators must not have any leading whitespace and must have
NOTE exactly three hyphen characters. The lines immediately before and after the
separator must not be same comment prefix.
Multi-document property files are often used in conjunction with activation properties
TIP
such as spring.config.activate.on-profile. See the next section for details.
Activation Properties
It is sometimes useful to only activate a given set of properties when certain conditions are met. For
example, you might have properties that are only relevant when a specific profile is active.
Property Note
on-profile A profile expression that must match for the document to be active.
on-cloud-platform The CloudPlatform that must be detected for the document to be active.
For example, the following specifies that the second document is only active when running on
Kubernetes, and only when either the “prod” or “staging” profiles are active:
83
Properties
myprop=always-set
#---
spring.config.activate.on-cloud-platform=kubernetes
spring.config.activate.on-profile=prod | staging
myotherprop=sometimes-set
Yaml
myprop:
"always-set"
---
spring:
config:
activate:
on-cloud-platform: "kubernetes"
on-profile: "prod | staging"
myotherprop: "sometimes-set"
Spring Boot does not provide any built-in support for encrypting property values, however, it does
provide the hook points necessary to modify values contained in the Spring Environment. The
EnvironmentPostProcessor interface allows you to manipulate the Environment before the application
starts. See Customize the Environment or ApplicationContext Before It Starts for details.
If you need a secure way to store credentials and passwords, the Spring Cloud Vault project
provides support for storing externalized configuration in HashiCorp Vault.
YAML is a superset of JSON and, as such, is a convenient format for specifying hierarchical
configuration data. The SpringApplication class automatically supports YAML as an alternative to
properties whenever you have the SnakeYAML library on your classpath.
YAML documents need to be converted from their hierarchical format to a flat structure that can be
used with the Spring Environment. For example, consider the following YAML document:
84
environments:
dev:
url: "https://dev.example.com"
name: "Developer Setup"
prod:
url: "https://another.example.com"
name: "My Cool App"
In order to access these properties from the Environment, they would be flattened as follows:
environments.dev.url=https://dev.example.com
environments.dev.name=Developer Setup
environments.prod.url=https://another.example.com
environments.prod.name=My Cool App
Likewise, YAML lists also need to be flattened. They are represented as property keys with [index]
dereferencers. For example, consider the following YAML:
my:
servers:
- "dev.example.com"
- "another.example.com"
my.servers[0]=dev.example.com
my.servers[1]=another.example.com
Properties that use the [index] notation can be bound to Java List or Set objects using
TIP Spring Boot’s Binder class. For more details see the “Type-safe Configuration
Properties” section below.
Spring Framework provides two convenient classes that can be used to load YAML documents. The
YamlPropertiesFactoryBean loads YAML as Properties and the YamlMapFactoryBean loads YAML as a
Map.
You can also use the YamlPropertySourceLoader class if you want to load YAML as a Spring
PropertySource.
85
7.2.6. Configuring Random Values
The RandomValuePropertySource is useful for injecting random values (for example, into secrets or
test cases). It can produce integers, longs, uuids, or strings, as shown in the following example:
Properties
my.secret=${random.value}
my.number=${random.int}
my.bignumber=${random.long}
my.uuid=${random.uuid}
my.number-less-than-ten=${random.int(10)}
my.number-in-range=${random.int[1024,65536]}
Yaml
my:
secret: "${random.value}"
number: "${random.int}"
bignumber: "${random.long}"
uuid: "${random.uuid}"
number-less-than-ten: "${random.int(10)}"
number-in-range: "${random.int[1024,65536]}"
The random.int* syntax is OPEN value (,max) CLOSE where the OPEN,CLOSE are any character and
value,max are integers. If max is provided, then value is the minimum value and max is the maximum
value (exclusive).
Spring Boot supports setting a prefix for environment properties. This is useful if the system
environment is shared by multiple Spring Boot applications with different configuration
requirements. The prefix for system environment properties can be set directly on
SpringApplication.
For example, if you set the prefix to input, a property such as remote.timeout will also be resolved as
input.remote.timeout in the system environment.
TIP See also the differences between @Value and type-safe configuration properties.
86
JavaBean Properties Binding
It is possible to bind a bean declaring standard JavaBean properties as shown in the following
example:
Java
import java.net.InetAddress;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("my.service")
public class MyProperties {
87
public String getUsername() {
return this.username;
}
88
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import java.net.InetAddress
@ConfigurationProperties("my.service")
class MyProperties {
class Security {
• my.service.security.password.
89
Such arrangement relies on a default empty constructor and getters and setters are
usually mandatory, since binding is through standard Java Beans property
descriptors, just like in Spring MVC. A setter may be omitted in the following cases:
• Maps, as long as they are initialized, need a getter but not necessarily a setter,
since they can be mutated by the binder.
• Collections and arrays can be accessed either through an index (typically with
YAML) or by using a single comma-separated value (properties). In the latter
case, a setter is mandatory. We recommend to always add a setter for such types.
If you initialize a collection, make sure it is not immutable (as in the preceding
NOTE example).
• If nested POJO properties are initialized (like the Security field in the preceding
example), a setter is not required. If you want the binder to create the instance
on the fly by using its default constructor, you need a setter.
Some people use Project Lombok to add getters and setters automatically. Make sure
that Lombok does not generate any particular constructor for such a type, as it is
used automatically by the container to instantiate the object.
Finally, only standard Java Bean properties are considered and binding on static
properties is not supported.
Constructor Binding
The example in the previous section can be rewritten in an immutable fashion as shown in the
following example:
Java
import java.net.InetAddress;
import java.util.List;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.bind.DefaultValue;
@ConfigurationProperties("my.service")
public class MyProperties {
90
this.security = security;
}
91
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.context.properties.bind.DefaultValue
import java.net.InetAddress
@ConfigurationProperties("my.service")
class MyProperties(val enabled: Boolean, val remoteAddress: InetAddress,
val security: Security) {
In this setup, the presence of a single parameterized constructor implies that constructor binding
should be used. This means that the binder will find a constructor with the parameters that you
wish to have bound. If your class has multiple constructors, the @ConstructorBinding annotation can
be used to specify which constructor to use for constructor binding. To opt out of constructor
binding for a class with a single parameterized constructor, the constructor must be annotated with
@Autowired. Constructor binding can be used with records. Unless your record has multiple
constructors, there is no need to use @ConstructorBinding.
Nested members of a constructor bound class (such as Security in the example above) will also be
bound through their constructor.
Default values can be specified using @DefaultValue on constructor parameters and record
components. The conversion service will be applied to coerce the annotation’s String value to the
target type of a missing property.
Referring to the previous example, if no properties are bound to Security, the MyProperties instance
will contain a null value for security. To make it contain a non-null instance of Security even when
no properties are bound to it (when using Kotlin, this will require the username and password
parameters of Security to be declared as nullable as they do not have default values), use an empty
@DefaultValue annotation:
Java
92
Kotlin
To use constructor binding in a native image the class must be compiled with
NOTE -parameters. This will happen automatically if you use Spring Boot’s Gradle plugin
or if you use Maven and spring-boot-starter-parent.
Spring Boot provides infrastructure to bind @ConfigurationProperties types and register them as
beans. You can either enable configuration properties on a class-by-class basis or enable
configuration property scanning that works in a similar manner to component scanning.
Sometimes, classes annotated with @ConfigurationProperties might not be suitable for scanning, for
example, if you’re developing your own auto-configuration or you want to enable them
conditionally. In these cases, specify the list of types to process using the
@EnableConfigurationProperties annotation. This can be done on any @Configuration class, as shown
in the following example:
93
Java
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(SomeProperties.class)
public class MyConfiguration {
Kotlin
import org.springframework.boot.context.properties.EnableConfigurationProperties
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(SomeProperties::class)
class MyConfiguration
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("some.properties")
public class SomeProperties {
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
@ConfigurationProperties("some.properties")
class SomeProperties
94
Java
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.context.properties.ConfigurationPropertiesScan;
@SpringBootApplication
@ConfigurationPropertiesScan({ "com.example.app", "com.example.another" })
public class MyApplication {
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.context.properties.ConfigurationPropertiesScan
@SpringBootApplication
@ConfigurationPropertiesScan("com.example.app", "com.example.another")
class MyApplication
We recommend that @ConfigurationProperties only deal with the environment and, in particular,
does not inject other beans from the context. For corner cases, setter injection can be used or any of
the *Aware interfaces provided by the framework (such as EnvironmentAware if you need access to the
Environment). If you still want to inject other beans using the constructor, the configuration
properties bean must be annotated with @Component and use JavaBean-based property binding.
This style of configuration works particularly well with the SpringApplication external YAML
configuration, as shown in the following example:
95
my:
service:
remote-address: 192.168.1.1
security:
username: "admin"
roles:
- "USER"
- "ADMIN"
To work with @ConfigurationProperties beans, you can inject them in the same way as any other
bean, as shown in the following example:
Java
import org.springframework.stereotype.Service;
@Service
public class MyService {
// ...
96
Kotlin
import org.springframework.stereotype.Service
@Service
class MyService(val properties: MyProperties) {
fun openConnection() {
val server = Server(properties.remoteAddress)
server.start()
// ...
}
// ...
Using @ConfigurationProperties also lets you generate metadata files that can be used
TIP
by IDEs to offer auto-completion for your own keys. See the appendix for details.
Third-party Configuration
As well as using @ConfigurationProperties to annotate a class, you can also use it on public @Bean
methods. Doing so can be particularly useful when you want to bind properties to third-party
components that are outside of your control.
To configure a bean from the Environment properties, add @ConfigurationProperties to its bean
registration, as shown in the following example:
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class ThirdPartyConfiguration {
@Bean
@ConfigurationProperties(prefix = "another")
public AnotherComponent anotherComponent() {
return new AnotherComponent();
}
97
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class ThirdPartyConfiguration {
@Bean
@ConfigurationProperties(prefix = "another")
fun anotherComponent(): AnotherComponent = AnotherComponent()
Any JavaBean property defined with the another prefix is mapped onto that AnotherComponent bean
in manner similar to the preceding SomeProperties example.
Relaxed Binding
Spring Boot uses some relaxed rules for binding Environment properties to @ConfigurationProperties
beans, so there does not need to be an exact match between the Environment property name and the
bean property name. Common examples where this is useful include dash-separated environment
properties (for example, context-path binds to contextPath), and capitalized environment properties
(for example, PORT binds to port).
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties(prefix = "my.main-project.person")
public class MyPersonProperties {
98
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
@ConfigurationProperties(prefix = "my.main-project.person")
class MyPersonProperties {
With the preceding code, the following properties names can all be used:
Property Note
my.main- Kebab case, which is recommended for use in .properties and YAML files.
project.person.fir
st-name
my.main- Standard camel case syntax.
project.person.fir
stName
my.main- Underscore notation, which is an alternative format for use in .properties and
project.person.fir YAML files.
st_name
MY_MAINPROJECT_PER Upper case format, which is recommended when using system environment
SON_FIRSTNAME variables.
The prefix value for the annotation must be in kebab case (lowercase and separated
NOTE
by -, such as my.main-project.person).
Properties Files Camel case, kebab case, or underscore Standard list syntax using [ ] or
notation comma-separated values
YAML Files Camel case, kebab case, or underscore Standard YAML list syntax or comma-
notation separated values
System properties Camel case, kebab case, or underscore Standard list syntax using [ ] or
notation comma-separated values
We recommend that, when possible, properties are stored in lower-case kebab format,
TIP
such as my.person.first-name=Rod.
99
Binding Maps
When binding to Map properties you may need to use a special bracket notation so that the original
key value is preserved. If the key is not surrounded by [], any characters that are not alpha-
numeric, - or . are removed.
Properties
my.map.[/key1]=value1
my.map.[/key2]=value2
my.map./key3=value3
Yaml
my:
map:
"[/key1]": "value1"
"[/key2]": "value2"
"/key3": "value3"
For YAML files, the brackets need to be surrounded by quotes for the keys to be
NOTE
parsed properly.
The properties above will bind to a Map with /key1, /key2 and key3 as the keys in the map. The slash
has been removed from key3 because it was not surrounded by square brackets.
When binding to scalar values, keys with . in them do not need to be surrounded by []. Scalar
values include enums and all types in the java.lang package except for Object. Binding a.b=c to
Map<String, String> will preserve the . in the key and return a Map with the entry {"a.b"="c"}. For
any other types you need to use the bracket notation if your key contains a .. For example, binding
a.b=c to Map<String, Object> will return a Map with the entry {"a"={"b"="c"}} whereas [a.b]=c will
return a Map with the entry {"a.b"="c"}.
Most operating systems impose strict rules around the names that can be used for environment
variables. For example, Linux shell variables can contain only letters (a to z or A to Z), numbers (0 to
9) or the underscore character (_). By convention, Unix shell variables will also have their names in
UPPERCASE.
Spring Boot’s relaxed binding rules are, as much as possible, designed to be compatible with these
naming restrictions.
To convert a property name in the canonical-form to an environment variable name you can follow
these rules:
100
• Remove any dashes (-).
• Convert to uppercase.
Environment variables can also be used when binding to object lists. To bind to a List, the element
number should be surrounded with underscores in the variable name.
For example, the configuration property my.service[0].other would use an environment variable
named MY_SERVICE_0_OTHER.
When lists are configured in more than one place, overriding works by replacing the entire list.
For example, assume a MyPojo object with name and description attributes that are null by default.
The following example exposes a list of MyPojo objects from MyProperties:
Java
import java.util.ArrayList;
import java.util.List;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("my")
public class MyProperties {
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
@ConfigurationProperties("my")
class MyProperties {
101
Properties
my.list[0].name=my name
my.list[0].description=my description
#---
spring.config.activate.on-profile=dev
my.list[0].name=my another name
Yaml
my:
list:
- name: "my name"
description: "my description"
---
spring:
config:
activate:
on-profile: "dev"
my:
list:
- name: "my another name"
If the dev profile is not active, MyProperties.list contains one MyPojo entry, as previously defined. If
the dev profile is enabled, however, the list still contains only one entry (with a name of my another
name and a description of null). This configuration does not add a second MyPojo instance to the list,
and it does not merge the items.
When a List is specified in multiple profiles, the one with the highest priority (and only that one) is
used. Consider the following example:
Properties
my.list[0].name=my name
my.list[0].description=my description
my.list[1].name=another name
my.list[1].description=another description
#---
spring.config.activate.on-profile=dev
my.list[0].name=my another name
102
Yaml
my:
list:
- name: "my name"
description: "my description"
- name: "another name"
description: "another description"
---
spring:
config:
activate:
on-profile: "dev"
my:
list:
- name: "my another name"
In the preceding example, if the dev profile is active, MyProperties.list contains one MyPojo entry
(with a name of my another name and a description of null). For YAML, both comma-separated lists
and YAML lists can be used for completely overriding the contents of the list.
For Map properties, you can bind with property values drawn from multiple sources. However, for
the same property in multiple sources, the one with the highest priority is used. The following
example exposes a Map<String, MyPojo> from MyProperties:
Java
import java.util.LinkedHashMap;
import java.util.Map;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("my")
public class MyProperties {
103
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
@ConfigurationProperties("my")
class MyProperties {
Properties
my.map.key1.name=my name 1
my.map.key1.description=my description 1
#---
spring.config.activate.on-profile=dev
my.map.key1.name=dev name 1
my.map.key2.name=dev name 2
my.map.key2.description=dev description 2
Yaml
my:
map:
key1:
name: "my name 1"
description: "my description 1"
---
spring:
config:
activate:
on-profile: "dev"
my:
map:
key1:
name: "dev name 1"
key2:
name: "dev name 2"
description: "dev description 2"
If the dev profile is not active, MyProperties.map contains one entry with key key1 (with a name of my
name 1 and a description of my description 1). If the dev profile is enabled, however, map contains
two entries with keys key1 (with a name of dev name 1 and a description of my description 1) and
key2 (with a name of dev name 2 and a description of dev description 2).
104
The preceding merging rules apply to properties from all property sources, and not
NOTE
just files.
Properties Conversion
Spring Boot attempts to coerce the external application properties to the right type when it binds to
the @ConfigurationProperties beans. If you need custom type conversion, you can provide a
ConversionService bean (with a bean named conversionService) or custom property editors (through
a CustomEditorConfigurer bean) or custom Converters (with bean definitions annotated as
@ConfigurationPropertiesBinding).
As this bean is requested very early during the application lifecycle, make sure to
limit the dependencies that your ConversionService is using. Typically, any
dependency that you require may not be fully initialized at creation time. You may
NOTE
want to rename your custom ConversionService if it is not required for
configuration keys coercion and only rely on custom converters qualified with
@ConfigurationPropertiesBinding.
Converting Durations
Spring Boot has dedicated support for expressing durations. If you expose a java.time.Duration
property, the following formats in application properties are available:
• A regular long representation (using milliseconds as the default unit unless a @DurationUnit has
been specified)
• A more readable format where the value and the unit are coupled (10s means 10 seconds)
105
Java
import java.time.Duration;
import java.time.temporal.ChronoUnit;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.convert.DurationUnit;
@ConfigurationProperties("my")
public class MyProperties {
@DurationUnit(ChronoUnit.SECONDS)
private Duration sessionTimeout = Duration.ofSeconds(30);
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.convert.DurationUnit
import java.time.Duration
import java.time.temporal.ChronoUnit
@ConfigurationProperties("my")
class MyProperties {
@DurationUnit(ChronoUnit.SECONDS)
var sessionTimeout = Duration.ofSeconds(30)
106
To specify a session timeout of 30 seconds, 30, PT30S and 30s are all equivalent. A read timeout of
500ms can be specified in any of the following form: 500, PT0.5S and 500ms.
You can also use any of the supported units. These are:
• ns for nanoseconds
• us for microseconds
• ms for milliseconds
• s for seconds
• m for minutes
• h for hours
• d for days
The default unit is milliseconds and can be overridden using @DurationUnit as illustrated in the
sample above.
If you prefer to use constructor binding, the same properties can be exposed, as shown in the
following example:
107
Java
import java.time.Duration;
import java.time.temporal.ChronoUnit;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.bind.DefaultValue;
import org.springframework.boot.convert.DurationUnit;
@ConfigurationProperties("my")
public class MyProperties {
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.context.properties.bind.DefaultValue
import org.springframework.boot.convert.DurationUnit
import java.time.Duration
import java.time.temporal.ChronoUnit
@ConfigurationProperties("my")
class MyProperties(@param:DurationUnit(ChronoUnit.SECONDS) @param:DefaultValue("30s")
val sessionTimeout: Duration,
@param:DefaultValue("1000ms") val readTimeout: Duration)
If you are upgrading a Long property, make sure to define the unit (using
TIP @DurationUnit) if it is not milliseconds. Doing so gives a transparent upgrade path
while supporting a much richer format.
108
Converting Periods
In addition to durations, Spring Boot can also work with java.time.Period type. The following
formats can be used in application properties:
• An regular int representation (using days as the default unit unless a @PeriodUnit has been
specified)
• A simpler format where the value and the unit pairs are coupled (1y3d means 1 year and 3 days)
• y for years
• m for months
• w for weeks
• d for days
Spring Framework has a DataSize value type that expresses a size in bytes. If you expose a DataSize
property, the following formats in application properties are available:
• A regular long representation (using bytes as the default unit unless a @DataSizeUnit has been
specified)
• A more readable format where the value and the unit are coupled (10MB means 10 megabytes)
109
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.convert.DataSizeUnit;
import org.springframework.util.unit.DataSize;
import org.springframework.util.unit.DataUnit;
@ConfigurationProperties("my")
public class MyProperties {
@DataSizeUnit(DataUnit.MEGABYTES)
private DataSize bufferSize = DataSize.ofMegabytes(2);
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.convert.DataSizeUnit
import org.springframework.util.unit.DataSize
import org.springframework.util.unit.DataUnit
@ConfigurationProperties("my")
class MyProperties {
@DataSizeUnit(DataUnit.MEGABYTES)
var bufferSize = DataSize.ofMegabytes(2)
110
To specify a buffer size of 10 megabytes, 10 and 10MB are equivalent. A size threshold of 256 bytes
can be specified as 256 or 256B.
You can also use any of the supported units. These are:
• B for bytes
• KB for kilobytes
• MB for megabytes
• GB for gigabytes
• TB for terabytes
The default unit is bytes and can be overridden using @DataSizeUnit as illustrated in the sample
above.
If you prefer to use constructor binding, the same properties can be exposed, as shown in the
following example:
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.bind.DefaultValue;
import org.springframework.boot.convert.DataSizeUnit;
import org.springframework.util.unit.DataSize;
import org.springframework.util.unit.DataUnit;
@ConfigurationProperties("my")
public class MyProperties {
111
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.context.properties.bind.DefaultValue
import org.springframework.boot.convert.DataSizeUnit
import org.springframework.util.unit.DataSize
import org.springframework.util.unit.DataUnit
@ConfigurationProperties("my")
class MyProperties(@param:DataSizeUnit(DataUnit.MEGABYTES) @param:DefaultValue("2MB")
val bufferSize: DataSize,
@param:DefaultValue("512B") val sizeThreshold: DataSize)
If you are upgrading a Long property, make sure to define the unit (using
TIP @DataSizeUnit) if it is not bytes. Doing so gives a transparent upgrade path while
supporting a much richer format.
@ConfigurationProperties Validation
Spring Boot attempts to validate @ConfigurationProperties classes whenever they are annotated
with Spring’s @Validated annotation. You can use JSR-303 jakarta.validation constraint annotations
directly on your configuration class. To do so, ensure that a compliant JSR-303 implementation is on
your classpath and then add constraint annotations to your fields, as shown in the following
example:
112
Java
import java.net.InetAddress;
import jakarta.validation.constraints.NotNull;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.validation.annotation.Validated;
@ConfigurationProperties("my.service")
@Validated
public class MyProperties {
@NotNull
private InetAddress remoteAddress;
Kotlin
import jakarta.validation.constraints.NotNull
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.validation.annotation.Validated
import java.net.InetAddress
@ConfigurationProperties("my.service")
@Validated
class MyProperties {
You can also trigger validation by annotating the @Bean method that creates the
TIP
configuration properties with @Validated.
To ensure that validation is always triggered for nested properties, even when no properties are
found, the associated field must be annotated with @Valid. The following example builds on the
preceding MyProperties example:
113
Java
import java.net.InetAddress;
import jakarta.validation.Valid;
import jakarta.validation.constraints.NotEmpty;
import jakarta.validation.constraints.NotNull;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.validation.annotation.Validated;
@ConfigurationProperties("my.service")
@Validated
public class MyProperties {
@NotNull
private InetAddress remoteAddress;
@Valid
private final Security security = new Security();
@NotEmpty
private String username;
114
Kotlin
import jakarta.validation.Valid
import jakarta.validation.constraints.NotEmpty
import jakarta.validation.constraints.NotNull
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.validation.annotation.Validated
import java.net.InetAddress
@ConfigurationProperties("my.service")
@Validated
class MyProperties {
@Valid
val security = Security()
class Security {
@NotEmpty
var username: String? = null
You can also add a custom Spring Validator by creating a bean definition called
configurationPropertiesValidator. The @Bean method should be declared static. The configuration
properties validator is created very early in the application’s lifecycle, and declaring the @Bean
method as static lets the bean be created without having to instantiate the @Configuration class.
Doing so avoids any problems that may be caused by early instantiation.
The @Value annotation is a core container feature, and it does not provide the same features as type-
safe configuration properties. The following table summarizes the features that are supported by
@ConfigurationProperties and @Value:
115
Feature @ConfigurationProperti @Value
es
If you do want to use @Value, we recommend that you refer to property names using
their canonical form (kebab-case using only lowercase letters). This will allow
Spring Boot to use the same logic as it does when relaxed binding
@ConfigurationProperties.
NOTE
For example, @Value("${demo.item-price}") will pick up demo.item-price and
demo.itemPrice forms from the application.properties file, as well as DEMO_ITEMPRICE
from the system environment. If you used @Value("${demo.itemPrice}") instead,
demo.item-price and DEMO_ITEMPRICE would not be considered.
If you define a set of configuration keys for your own components, we recommend you group them
in a POJO annotated with @ConfigurationProperties. Doing so will provide you with structured,
type-safe object that you can inject into your own beans.
SpEL expressions from application property files are not processed at time of parsing these files and
populating the environment. However, it is possible to write a SpEL expression in @Value. If the
value of a property from an application property file is a SpEL expression, it will be evaluated when
consumed through @Value.
7.3. Profiles
Spring Profiles provide a way to segregate parts of your application configuration and make it be
available only in certain environments. Any @Component, @Configuration or @ConfigurationProperties
can be marked with @Profile to limit when it is loaded, as shown in the following example:
Java
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;
@Configuration(proxyBeanMethods = false)
@Profile("production")
public class ProductionConfiguration {
// ...
116
Kotlin
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Profile
@Configuration(proxyBeanMethods = false)
@Profile("production")
class ProductionConfiguration {
// ...
You can use a spring.profiles.active Environment property to specify which profiles are active. You
can specify the property in any of the ways described earlier in this chapter. For example, you
could include it in your application.properties, as shown in the following example:
Properties
spring.profiles.active=dev,hsqldb
Yaml
spring:
profiles:
active: "dev,hsqldb"
You could also specify it on the command line by using the following switch:
--spring.profiles.active=dev,hsqldb.
If no profile is active, a default profile is enabled. The name of the default profile is default and it
can be tuned using the spring.profiles.default Environment property, as shown in the following
example:
Properties
spring.profiles.default=none
117
Yaml
spring:
profiles:
default: "none"
Properties
Yaml
The spring.profiles.active property follows the same ordering rules as other properties: The
highest PropertySource wins. This means that you can specify active profiles in
application.properties and then replace them by using the command line switch.
Sometimes, it is useful to have properties that add to the active profiles rather than replace them.
The spring.profiles.include property can be used to add active profiles on top of those activated by
the spring.profiles.active property. The SpringApplication entry point also has a Java API for
setting additional profiles. See the setAdditionalProfiles() method in SpringApplication.
For example, when an application with the following properties is run, the common and local
profiles will be activated even when it runs using the --spring.profiles.active switch:
118
Properties
spring.profiles.include[0]=common
spring.profiles.include[1]=local
Yaml
spring:
profiles:
include:
- "common"
- "local"
Profile groups, which are described in the next section can also be used to add active profiles if a
given profile is active.
Occasionally the profiles that you define and use in your application are too fine-grained and
become cumbersome to use. For example, you might have proddb and prodmq profiles that you use to
enable database and messaging features independently.
To help with this, Spring Boot lets you define profile groups. A profile group allows you to define a
logical name for a related group of profiles.
For example, we can create a production group that consists of our proddb and prodmq profiles.
Properties
spring.profiles.group.production[0]=proddb
spring.profiles.group.production[1]=prodmq
Yaml
spring:
profiles:
group:
production:
- "proddb"
- "prodmq"
119
7.3.3. Programmatically Setting Profiles
7.4. Logging
Spring Boot uses Commons Logging for all internal logging but leaves the underlying log
implementation open. Default configurations are provided for Java Util Logging, Log4j2, and
Logback. In each case, loggers are pre-configured to use console output with optional file output
also available.
By default, if you use the “Starters”, Logback is used for logging. Appropriate Logback routing is
also included to ensure that dependent libraries that use Java Util Logging, Commons Logging,
Log4J, or SLF4J all work correctly.
There are a lot of logging frameworks available for Java. Do not worry if the above list
TIP seems confusing. Generally, you do not need to change your logging dependencies and
the Spring Boot defaults work just fine.
When you deploy your application to a servlet container or application server, logging
performed with the Java Util Logging API is not routed into your application’s logs.
TIP
This prevents logging performed by the container or other applications that have been
deployed to it from appearing in your application’s logs.
The default log output from Spring Boot resembles the following example:
120
2023-05-05T05:10:37.703Z INFO 1280 --- [ main] o.s.b.d.f.s.MyApplication
: Starting MyApplication using Java 17.0.7 with PID 1280 (/opt/apps/myapp.jar started
by myuser in /opt/apps/)
2023-05-05T05:10:37.715Z INFO 1280 --- [ main] o.s.b.d.f.s.MyApplication
: No active profile set, falling back to 1 default profile: "default"
2023-05-05T05:10:41.940Z INFO 1280 --- [ main]
o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat initialized with port(s): 8080
(http)
2023-05-05T05:10:41.953Z INFO 1280 --- [ main]
o.apache.catalina.core.StandardService : Starting service [Tomcat]
2023-05-05T05:10:41.953Z INFO 1280 --- [ main]
o.apache.catalina.core.StandardEngine : Starting Servlet engine: [Apache
Tomcat/10.1.8]
2023-05-05T05:10:42.431Z INFO 1280 --- [ main]
o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded
WebApplicationContext
2023-05-05T05:10:42.434Z INFO 1280 --- [ main]
w.s.c.ServletWebServerApplicationContext : Root WebApplicationContext: initialization
completed in 4443 ms
2023-05-05T05:10:44.046Z INFO 1280 --- [ main]
o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port(s): 8080 (http) with
context path ''
2023-05-05T05:10:44.064Z INFO 1280 --- [ main] o.s.b.d.f.s.MyApplication
: Started MyApplication in 8.647 seconds (process running for 10.068)
• Process ID.
• Thread name: Enclosed in square brackets (may be truncated for console output).
• Logger name: This is usually the source class name (often abbreviated).
The default log configuration echoes messages to the console as they are written. By default, ERROR
-level, WARN-level, and INFO-level messages are logged. You can also enable a “debug” mode by
starting your application with a --debug flag.
121
NOTE You can also specify debug=true in your application.properties.
When the debug mode is enabled, a selection of core loggers (embedded container, Hibernate, and
Spring Boot) are configured to output more information. Enabling the debug mode does not
configure your application to log all messages with DEBUG level.
Alternatively, you can enable a “trace” mode by starting your application with a --trace flag (or
trace=true in your application.properties). Doing so enables trace logging for a selection of core
loggers (embedded container, Hibernate schema generation, and the whole Spring portfolio).
Color-coded Output
If your terminal supports ANSI, color output is used to aid readability. You can set
spring.output.ansi.enabled to a supported value to override the auto-detection.
Color coding is configured by using the %clr conversion word. In its simplest form, the converter
colors the output according to the log level, as shown in the following example:
%clr(%5p)
Level Color
FATAL Red
ERROR Red
WARN Yellow
INFO Green
DEBUG Green
TRACE Green
Alternatively, you can specify the color or style that should be used by providing it as an option to
the conversion. For example, to make the text yellow, use the following setting:
%clr(%d{yyyy-MM-dd'T'HH:mm:ss.SSSXXX}){yellow}
• blue
• cyan
• faint
• green
• magenta
• red
122
• yellow
By default, Spring Boot logs only to the console and does not write log files. If you want to write log
files in addition to the console output, you need to set a logging.file.name or logging.file.path
property (for example, in your application.properties).
The following table shows how the logging.* properties can be used together:
Specific file (none) my.log Writes to the specified log file. Names can be an exact
location or relative to the current directory.
(none) Specific /var/log Writes spring.log to the specified directory. Names can
directory be an exact location or relative to the current directory.
Log files rotate when they reach 10 MB and, as with console output, ERROR-level, WARN-level, and INFO
-level messages are logged by default.
If you are using the Logback, it is possible to fine-tune log rotation settings using your
application.properties or application.yaml file. For all other logging system, you will need to
configure rotation settings directly yourself (for example, if you use Log4j2 then you could add a
log4j2.xml or log4j2-spring.xml file).
Name Description
logging.logback.rollingpolicy.file-name- The filename pattern used to create log archives.
pattern
logging.logback.rollingpolicy.clean-history- If log archive cleanup should occur when the
on-start application starts.
logging.logback.rollingpolicy.max-file-size The maximum size of log file before it is
archived.
logging.logback.rollingpolicy.total-size-cap The maximum amount of size log archives can
take before being deleted.
123
Name Description
logging.logback.rollingpolicy.max-history The maximum number of archive log files to
keep (defaults to 7).
All the supported logging systems can have the logger levels set in the Spring Environment (for
example, in application.properties) by using logging.level.<logger-name>=<level> where level is
one of TRACE, DEBUG, INFO, WARN, ERROR, FATAL, or OFF. The root logger can be configured by
using logging.level.root.
Properties
logging.level.root=warn
logging.level.org.springframework.web=debug
logging.level.org.hibernate=error
Yaml
logging:
level:
root: "warn"
org.springframework.web: "debug"
org.hibernate: "error"
It is also possible to set logging levels using environment variables. For example,
LOGGING_LEVEL_ORG_SPRINGFRAMEWORK_WEB=DEBUG will set org.springframework.web to DEBUG.
The above approach will only work for package level logging. Since relaxed binding
always converts environment variables to lowercase, it is not possible to configure
NOTE
logging for an individual class in this way. If you need to configure logging for a
class, you can use the SPRING_APPLICATION_JSON variable.
It is often useful to be able to group related loggers together so that they can all be configured at the
same time. For example, you might commonly change the logging levels for all Tomcat related
loggers, but you can not easily remember top level packages.
To help with this, Spring Boot allows you to define logging groups in your Spring Environment. For
example, here is how you could define a “tomcat” group by adding it to your
application.properties:
124
Properties
logging.group.tomcat=org.apache.catalina,org.apache.coyote,org.apache.tomcat
Yaml
logging:
group:
tomcat: "org.apache.catalina,org.apache.coyote,org.apache.tomcat"
Once defined, you can change the level for all the loggers in the group with a single line:
Properties
logging.level.tomcat=trace
Yaml
logging:
level:
tomcat: "trace"
Spring Boot includes the following pre-defined logging groups that can be used out-of-the-box:
Name Loggers
In order to release logging resources when your application terminates, a shutdown hook that will
trigger log system cleanup when the JVM exits is provided. This shutdown hook is registered
automatically unless your application is deployed as a war file. If your application has complex
context hierarchies the shutdown hook may not meet your needs. If it does not, disable the
shutdown hook and investigate the options provided directly by the underlying logging system. For
example, Logback offers context selectors which allow each Logger to be created in its own context.
You can use the logging.register-shutdown-hook property to disable the shutdown hook. Setting it to
false will disable the registration. You can set the property in your application.properties or
application.yaml file:
Properties
logging.register-shutdown-hook=false
125
Yaml
logging:
register-shutdown-hook: false
The various logging systems can be activated by including the appropriate libraries on the
classpath and can be further customized by providing a suitable configuration file in the root of the
classpath or in a location specified by the following Spring Environment property: logging.config.
You can force Spring Boot to use a particular logging system by using the
org.springframework.boot.logging.LoggingSystem system property. The value should be the fully
qualified class name of a LoggingSystem implementation. You can also disable Spring Boot’s logging
configuration entirely by using a value of none.
When possible, we recommend that you use the -spring variants for your logging
configuration (for example, logback-spring.xml rather than logback.xml). If you use
NOTE
standard configuration locations, Spring cannot completely control log
initialization.
There are known classloading issues with Java Util Logging that cause
WARNING problems when running from an 'executable jar'. We recommend that you
avoid it when running from an 'executable jar' if at all possible.
To help with the customization, some other properties are transferred from the Spring Environment
to System properties, as described in the following table:
126
Spring Environment System Property Comments
logging.file.path LOG_PATH If defined, it is used in the
default log configuration.
logging.pattern.console CONSOLE_LOG_PATTERN The log pattern to use on the
console (stdout).
logging.pattern.dateformat LOG_DATEFORMAT_PATTERN Appender pattern for log date
format.
logging.charset.console CONSOLE_LOG_CHARSET The charset to use for console
logging.
logging.threshold.console CONSOLE_LOG_THRESHOLD The log level threshold to use
for console logging.
logging.pattern.file FILE_LOG_PATTERN The log pattern to use in a file
(if LOG_FILE is enabled).
logging.charset.file FILE_LOG_CHARSET The charset to use for file
logging (if LOG_FILE is enabled).
logging.threshold.file FILE_LOG_THRESHOLD The log level threshold to use
for file logging.
logging.pattern.level LOG_LEVEL_PATTERN The format to use when
rendering the log level (default
%5p).
PID PID The current process ID
(discovered if possible and
when not already defined as an
OS environment variable).
All the supported logging systems can consult System properties when parsing their configuration
127
files. See the default configurations in spring-boot.jar for examples:
• Logback
• Log4j 2
If you want to use a placeholder in a logging property, you should use Spring Boot’s
syntax and not the syntax of the underlying framework. Notably, if you use Logback,
TIP
you should use : as the delimiter between a property name and its default value and
not use :-.
You can add MDC and other ad-hoc content to log lines by overriding only the
LOG_LEVEL_PATTERN (or logging.pattern.level with Logback). For example, if you use
logging.pattern.level=user:%X{user} %5p, then the default log format contains an MDC
entry for "user", if it exists, as shown in the following example.
TIP
Spring Boot includes a number of extensions to Logback that can help with advanced configuration.
You can use these extensions in your logback-spring.xml configuration file.
Because the standard logback.xml configuration file is loaded too early, you cannot
NOTE use extensions in it. You need to either use logback-spring.xml or define a
logging.config property.
Profile-specific Configuration
The <springProfile> tag lets you optionally include or exclude sections of configuration based on
the active Spring profiles. Profile sections are supported anywhere within the <configuration>
element. Use the name attribute to specify which profile accepts the configuration. The
<springProfile> tag can contain a profile name (for example staging) or a profile expression. A
profile expression allows for more complicated profile logic to be expressed, for example
128
production & (eu-central | eu-west). Check the Spring Framework reference guide for more
details. The following listing shows three sample profiles:
<springProfile name="staging">
<!-- configuration to be enabled when the "staging" profile is active -->
</springProfile>
<springProfile name="!production">
<!-- configuration to be enabled when the "production" profile is not active -->
</springProfile>
Environment Properties
The <springProperty> tag lets you expose properties from the Spring Environment for use within
Logback. Doing so can be useful if you want to access values from your application.properties file
in your Logback configuration. The tag works in a similar way to Logback’s standard <property> tag.
However, rather than specifying a direct value, you specify the source of the property (from the
Environment). If you need to store the property somewhere other than in local scope, you can use
the scope attribute. If you need a fallback value (in case the property is not set in the Environment),
you can use the defaultValue attribute. The following example shows how to expose properties for
use within Logback:
Spring Boot includes a number of extensions to Log4j2 that can help with advanced configuration.
You can use these extensions in any log4j2-spring.xml configuration file.
Because the standard log4j2.xml configuration file is loaded too early, you cannot
NOTE use extensions in it. You need to either use log4j2-spring.xml or define a
logging.config property.
129
The extensions supersede the Spring Boot support provided by Log4J. You should
NOTE make sure not to include the org.apache.logging.log4j:log4j-spring-boot module in
your build.
Profile-specific Configuration
The <SpringProfile> tag lets you optionally include or exclude sections of configuration based on
the active Spring profiles. Profile sections are supported anywhere within the <Configuration>
element. Use the name attribute to specify which profile accepts the configuration. The
<SpringProfile> tag can contain a profile name (for example staging) or a profile expression. A
profile expression allows for more complicated profile logic to be expressed, for example
production & (eu-central | eu-west). Check the Spring Framework reference guide for more
details. The following listing shows three sample profiles:
<SpringProfile name="staging">
<!-- configuration to be enabled when the "staging" profile is active -->
</SpringProfile>
<SpringProfile name="!production">
<!-- configuration to be enabled when the "production" profile is not active -->
</SpringProfile>
If you want to refer to properties from your Spring Environment within your Log4j2 configuration
you can use spring: prefixed lookups. Doing so can be useful if you want to access values from your
application.properties file in your Log4j2 configuration.
The following example shows how to set a Log4j2 property named applicationName that reads
spring.application.name from the Spring Environment:
<Properties>
<Property name="applicationName">${spring:spring.application.name}</Property>
</Properties>
NOTE The lookup key should be specified in kebab case (such as my.property-name).
Log4j2 supports a number of System Properties that can be used to configure various items. For
example, the log4j2.skipJansi system property can be used to configure if the ConsoleAppender will
try to use a Jansi output stream on Windows.
130
All system properties that are loaded after the Log4j2 initialization can be obtained from the Spring
Environment. For example, you could add log4j2.skipJansi=false to your application.properties file
to have the ConsoleAppender use Jansi on Windows.
System properties that are loaded during early Log4j2 initialization cannot
reference the Spring Environment. For example, the property Log4j2 uses to
WARNING
allow the default Log4j2 implementation to be chosen is used before the Spring
Environment is available.
7.5. Internationalization
Spring Boot supports localized messages so that your application can cater to users of different
language preferences. By default, Spring Boot looks for the presence of a messages resource bundle
at the root of the classpath.
The auto-configuration applies when the default properties file for the configured
resource bundle is available (messages.properties by default). If your resource
NOTE bundle contains only language-specific properties files, you are required to add the
default. If no properties file is found that matches any of the configured base
names, there will be no auto-configured MessageSource.
The basename of the resource bundle as well as several other attributes can be configured using
the spring.messages namespace, as shown in the following example:
Properties
spring.messages.basename=messages,config.i18n.messages
spring.messages.fallback-to-system-locale=false
Yaml
spring:
messages:
basename: "messages,config.i18n.messages"
fallback-to-system-locale: false
131
7.6. JSON
Spring Boot provides integration with three JSON mapping libraries:
• Gson
• Jackson
• JSON-B
7.6.1. Jackson
If you use Jackson to serialize and deserialize JSON data, you might want to write your own
JsonSerializer and JsonDeserializer classes. Custom serializers are usually registered with Jackson
through a module, but Spring Boot provides an alternative @JsonComponent annotation that makes it
easier to directly register Spring Beans.
132
Java
import java.io.IOException;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.ObjectCodec;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;
import org.springframework.boot.jackson.JsonComponent;
@JsonComponent
public class MyJsonComponent {
@Override
public void serialize(MyObject value, JsonGenerator jgen, SerializerProvider
serializers) throws IOException {
jgen.writeStartObject();
jgen.writeStringField("name", value.getName());
jgen.writeNumberField("age", value.getAge());
jgen.writeEndObject();
}
@Override
public MyObject deserialize(JsonParser jsonParser, DeserializationContext
ctxt) throws IOException {
ObjectCodec codec = jsonParser.getCodec();
JsonNode tree = codec.readTree(jsonParser);
String name = tree.get("name").textValue();
int age = tree.get("age").intValue();
return new MyObject(name, age);
}
133
Kotlin
import com.fasterxml.jackson.core.JsonGenerator
import com.fasterxml.jackson.core.JsonParser
import com.fasterxml.jackson.core.JsonProcessingException
import com.fasterxml.jackson.databind.DeserializationContext
import com.fasterxml.jackson.databind.JsonDeserializer
import com.fasterxml.jackson.databind.JsonNode
import com.fasterxml.jackson.databind.JsonSerializer
import com.fasterxml.jackson.databind.SerializerProvider
import org.springframework.boot.jackson.JsonComponent
import java.io.IOException
@JsonComponent
class MyJsonComponent {
All @JsonComponent beans in the ApplicationContext are automatically registered with Jackson.
Because @JsonComponent is meta-annotated with @Component, the usual component-scanning rules
apply.
Spring Boot also provides JsonObjectSerializer and JsonObjectDeserializer base classes that
provide useful alternatives to the standard Jackson versions when serializing objects. See
JsonObjectSerializer and JsonObjectDeserializer in the Javadoc for details.
134
Java
import java.io.IOException;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.ObjectCodec;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.SerializerProvider;
import org.springframework.boot.jackson.JsonComponent;
import org.springframework.boot.jackson.JsonObjectDeserializer;
import org.springframework.boot.jackson.JsonObjectSerializer;
@JsonComponent
public class MyJsonComponent {
@Override
protected void serializeObject(MyObject value, JsonGenerator jgen,
SerializerProvider provider)
throws IOException {
jgen.writeStringField("name", value.getName());
jgen.writeNumberField("age", value.getAge());
}
@Override
protected MyObject deserializeObject(JsonParser jsonParser,
DeserializationContext context, ObjectCodec codec,
JsonNode tree) throws IOException {
String name = nullSafeValue(tree.get("name"), String.class);
int age = nullSafeValue(tree.get("age"), Integer.class);
return new MyObject(name, age);
}
135
Kotlin
`object`
import com.fasterxml.jackson.core.JsonGenerator
import com.fasterxml.jackson.core.JsonParser
import com.fasterxml.jackson.core.ObjectCodec
import com.fasterxml.jackson.databind.DeserializationContext
import com.fasterxml.jackson.databind.JsonNode
import com.fasterxml.jackson.databind.SerializerProvider
import org.springframework.boot.jackson.JsonComponent
import org.springframework.boot.jackson.JsonObjectDeserializer
import org.springframework.boot.jackson.JsonObjectSerializer
import java.io.IOException
@JsonComponent
class MyJsonComponent {
Mixins
Jackson has support for mixins that can be used to mix additional annotations into those already
declared on a target class. Spring Boot’s Jackson auto-configuration will scan your application’s
packages for classes annotated with @JsonMixin and register them with the auto-configured
ObjectMapper. The registration is performed by Spring Boot’s JsonMixinModule.
7.6.2. Gson
Auto-configuration for Gson is provided. When Gson is on the classpath a Gson bean is automatically
136
configured. Several spring.gson.* configuration properties are provided for customizing the
configuration. To take more control, one or more GsonBuilderCustomizer beans can be used.
7.6.3. JSON-B
Auto-configuration for JSON-B is provided. When the JSON-B API and an implementation are on the
classpath a Jsonb bean will be automatically configured. The preferred JSON-B implementation is
Eclipse Yasson for which dependency management is provided.
If you have defined a custom Executor in the context, regular task execution (that is
@EnableAsync) will use it transparently but the Spring MVC support will not be
configured as it requires an AsyncTaskExecutor implementation (named
applicationTaskExecutor). Depending on your target arrangement, you could change
TIP your Executor into a ThreadPoolTaskExecutor or define both a ThreadPoolTaskExecutor
and an AsyncConfigurer wrapping your custom Executor.
The thread pool uses 8 core threads that can grow and shrink according to the load. Those default
settings can be fine-tuned using the spring.task.execution namespace, as shown in the following
example:
Properties
spring.task.execution.pool.max-size=16
spring.task.execution.pool.queue-capacity=100
spring.task.execution.pool.keep-alive=10s
Yaml
spring:
task:
execution:
pool:
max-size: 16
queue-capacity: 100
keep-alive: "10s"
This changes the thread pool to use a bounded queue so that when the queue is full (100 tasks), the
thread pool increases to maximum 16 threads. Shrinking of the pool is more aggressive as threads
are reclaimed when they are idle for 10 seconds (rather than 60 seconds by default).
137
A ThreadPoolTaskScheduler can also be auto-configured if need to be associated to scheduled task
execution (using @EnableScheduling for instance). The thread pool uses one thread by default and its
settings can be fine-tuned using the spring.task.scheduling namespace, as shown in the following
example:
Properties
spring.task.scheduling.thread-name-prefix=scheduling-
spring.task.scheduling.pool.size=2
Yaml
spring:
task:
scheduling:
thread-name-prefix: "scheduling-"
pool:
size: 2
Both a TaskExecutorBuilder bean and a TaskSchedulerBuilder bean are made available in the context
if a custom executor or scheduler needs to be created.
7.8. Testing
Spring Boot provides a number of utilities and annotations to help when testing your application.
Test support is provided by two modules: spring-boot-test contains core items, and spring-boot-
test-autoconfigure supports auto-configuration for tests.
Most developers use the spring-boot-starter-test “Starter”, which imports both Spring Boot test
modules as well as JUnit Jupiter, AssertJ, Hamcrest, and a number of other useful libraries.
If you have tests that use JUnit 4, JUnit 5’s vintage engine can be used to run them. To
use the vintage engine, add a dependency on junit-vintage-engine, as shown in the
following example:
<dependency>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
TIP <scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-core</artifactId>
</exclusion>
</exclusions>
</dependency>
138
hamcrest-core is excluded in favor of org.hamcrest:hamcrest that is part of spring-boot-starter-test.
The spring-boot-starter-test “Starter” (in the test scope) contains the following provided libraries:
• Spring Test & Spring Boot Test: Utilities and integration test support for Spring Boot
applications.
We generally find these common libraries to be useful when writing tests. If these libraries do not
suit your needs, you can add additional test dependencies of your own.
One of the major advantages of dependency injection is that it should make your code easier to unit
test. You can instantiate objects by using the new operator without even involving Spring. You can
also use mock objects instead of real dependencies.
Often, you need to move beyond unit testing and start integration testing (with a Spring
ApplicationContext). It is useful to be able to perform integration testing without requiring
deployment of your application or needing to connect to other infrastructure.
The Spring Framework includes a dedicated test module for such integration testing. You can
declare a dependency directly to org.springframework:spring-test or use the spring-boot-starter-
test “Starter” to pull it in transitively.
If you have not used the spring-test module before, you should start by reading the relevant
section of the Spring Framework reference documentation.
A Spring Boot application is a Spring ApplicationContext, so nothing very special has to be done to
test it beyond what you would normally do with a vanilla Spring context.
External properties, logging, and other features of Spring Boot are installed in the
NOTE
context by default only if you use SpringApplication to create it.
Spring Boot provides a @SpringBootTest annotation, which can be used as an alternative to the
standard spring-test @ContextConfiguration annotation when you need Spring Boot features. The
annotation works by creating the ApplicationContext used in your tests through SpringApplication.
In addition to @SpringBootTest a number of other annotations are also provided for testing more
139
specific slices of an application.
If you are using JUnit 4, do not forget to also add @RunWith(SpringRunner.class) to your
test, otherwise the annotations will be ignored. If you are using JUnit 5, there is no
TIP
need to add the equivalent @ExtendWith(SpringExtension.class) as @SpringBootTest and
the other @…Test annotations are already annotated with it.
By default, @SpringBootTest will not start a server. You can use the webEnvironment attribute of
@SpringBootTest to further refine how your tests run:
• NONE: Loads an ApplicationContext by using SpringApplication but does not provide any web
environment (mock or otherwise).
If your test is @Transactional, it rolls back the transaction at the end of each test
method by default. However, as using this arrangement with either RANDOM_PORT or
NOTE DEFINED_PORT implicitly provides a real servlet environment, the HTTP client and
server run in separate threads and, thus, in separate transactions. Any transaction
initiated on the server does not roll back in this case.
If Spring MVC is available, a regular MVC-based application context is configured. If you have only
Spring WebFlux, we will detect that and configure a WebFlux-based application context instead.
If both are present, Spring MVC takes precedence. If you want to test a reactive web application in
this scenario, you must set the spring.main.web-application-type property:
140
Java
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest(properties = "spring.main.web-application-type=reactive")
class MyWebFluxTests {
// ...
Kotlin
import org.springframework.boot.test.context.SpringBootTest
@SpringBootTest(properties = ["spring.main.web-application-type=reactive"])
class MyWebFluxTests {
// ...
If you are familiar with the Spring Test Framework, you may be used to using
@ContextConfiguration(classes=…) in order to specify which Spring @Configuration to load.
Alternatively, you might have often used nested @Configuration classes within your test.
When testing Spring Boot applications, this is often not required. Spring Boot’s @*Test annotations
search for your primary configuration automatically whenever you do not explicitly define one.
The search algorithm works up from the package that contains the test until it finds a class
annotated with @SpringBootApplication or @SpringBootConfiguration. As long as you structured your
code in a sensible way, your main configuration is usually found.
If you use a test annotation to test a more specific slice of your application, you
should avoid adding configuration settings that are specific to a particular area on
the main method’s application class.
If you want to customize the primary configuration, you can use a nested @TestConfiguration class.
Unlike a nested @Configuration class, which would be used instead of your application’s primary
configuration, a nested @TestConfiguration class is used in addition to your application’s primary
configuration.
141
Spring’s test framework caches application contexts between tests. Therefore, as
NOTE long as your tests share the same configuration (no matter how it is discovered), the
potentially time-consuming process of loading the context happens only once.
For example, the following is a very common code pattern for a typical Spring Boot application:
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import
org.springframework.boot.docs.using.structuringyourcode.locatingthemainclass.MyApplica
tion
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
In the example above, the main method doesn’t do anything other than delegate to
SpringApplication.run. It is, however, possible to have a more complex main method that applies
customizations before calling SpringApplication.run.
For example, here is an application that changes the banner mode and sets additional profiles:
142
Java
import org.springframework.boot.Banner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
Kotlin
import org.springframework.boot.Banner
import org.springframework.boot.runApplication
import org.springframework.boot.autoconfigure.SpringBootApplication
@SpringBootApplication
class MyApplication
Since customizations in the main method can affect the resulting ApplicationContext, it’s possible
that you might also want to use the main method to create the ApplicationContext used in your tests.
By default, @SpringBootTest will not call your main method, and instead the class itself is used
directly to create the ApplicationContext
If you want to change this behavior, you can change the useMainMethod attribute of @SpringBootTest
to UseMainMethod.ALWAYS or UseMainMethod.WHEN_AVAILABLE. When set to ALWAYS, the test will fail if no
main method can be found. When set to WHEN_AVAILABLE the main method will be used if it is available,
otherwise the standard loading mechanism will be used.
For example, the following test will invoke the main method of MyApplication in order to create the
ApplicationContext. If the main method sets additional profiles then those will be active when the
ApplicationContext starts.
143
Java
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.UseMainMethod;
@SpringBootTest(useMainMethod = UseMainMethod.ALWAYS)
class MyApplicationTests {
@Test
void exampleTest() {
// ...
}
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.UseMainMethod
import org.springframework.context.annotation.Import
@SpringBootTest(useMainMethod = UseMainMethod.ALWAYS)
class MyApplicationTests {
@Test
fun exampleTest() {
// ...
}
If your application uses component scanning (for example, if you use @SpringBootApplication or
@ComponentScan), you may find top-level configuration classes that you created only for specific tests
accidentally get picked up everywhere.
As we have seen earlier, @TestConfiguration can be used on an inner class of a test to customize the
primary configuration. When placed on a top-level class, @TestConfiguration indicates that classes
in src/test/java should not be picked up by scanning. You can then import that class explicitly
where it is required, as shown in the following example:
144
Java
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.annotation.Import;
@SpringBootTest
@Import(MyTestsConfiguration.class)
class MyTests {
@Test
void exampleTest() {
// ...
}
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.context.annotation.Import
@SpringBootTest
@Import(MyTestsConfiguration::class)
class MyTests {
@Test
fun exampleTest() {
// ...
}
If you directly use @ComponentScan (that is, not through @SpringBootApplication) you
NOTE
need to register the TypeExcludeFilter with it. See the Javadoc for details.
If your application expects arguments, you can have @SpringBootTest inject them using the args
attribute.
145
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest(args = "--app.test=one")
class MyApplicationArgumentTests {
@Test
void applicationArgumentsPopulated(@Autowired ApplicationArguments args) {
assertThat(args.getOptionNames()).containsOnly("app.test");
assertThat(args.getOptionValues("app.test")).containsOnly("one");
}
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.ApplicationArguments
import org.springframework.boot.test.context.SpringBootTest
@SpringBootTest(args = ["--app.test=one"])
class MyApplicationArgumentTests {
@Test
fun applicationArgumentsPopulated(@Autowired args: ApplicationArguments) {
assertThat(args.optionNames).containsOnly("app.test")
assertThat(args.getOptionValues("app.test")).containsOnly("one")
}
By default, @SpringBootTest does not start the server but instead sets up a mock environment for
testing web endpoints.
With Spring MVC, we can query our web endpoints using MockMvc or WebTestClient, as shown in the
following example:
146
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.web.reactive.server.WebTestClient;
import org.springframework.test.web.servlet.MockMvc;
@SpringBootTest
@AutoConfigureMockMvc
class MyMockMvcTests {
@Test
void testWithMockMvc(@Autowired MockMvc mvc) throws Exception {
mvc.perform(get("/")).andExpect(status().isOk()).andExpect(content().string("Hello
World"));
}
// If Spring WebFlux is on the classpath, you can drive MVC tests with a
WebTestClient
@Test
void testWithWebTestClient(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
147
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.test.web.reactive.server.WebTestClient
import org.springframework.test.web.reactive.server.expectBody
import org.springframework.test.web.servlet.MockMvc
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders
import org.springframework.test.web.servlet.result.MockMvcResultMatchers
@SpringBootTest
@AutoConfigureMockMvc
class MyMockMvcTests {
@Test
fun testWithMockMvc(@Autowired mvc: MockMvc) {
mvc.perform(MockMvcRequestBuilders.get("/")).andExpect(MockMvcResultMatchers.status().
isOk)
.andExpect(MockMvcResultMatchers.content().string("Hello World"))
}
// If Spring WebFlux is on the classpath, you can drive MVC tests with a
WebTestClient
@Test
fun testWithWebTestClient(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
If you want to focus only on the web layer and not start a complete
TIP
ApplicationContext, consider using @WebMvcTest instead.
With Spring WebFlux endpoints, you can use WebTestClient as shown in the following example:
148
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import
org.springframework.boot.test.autoconfigure.web.reactive.AutoConfigureWebTestClient;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.web.reactive.server.WebTestClient;
@SpringBootTest
@AutoConfigureWebTestClient
class MyMockWebTestClientTests {
@Test
void exampleTest(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import
org.springframework.boot.test.autoconfigure.web.reactive.AutoConfigureWebTestClient
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.test.web.reactive.server.WebTestClient
import org.springframework.test.web.reactive.server.expectBody
@SpringBootTest
@AutoConfigureWebTestClient
class MyMockWebTestClientTests {
@Test
fun exampleTest(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
149
Testing within a mocked environment is usually faster than running with a full servlet
container. However, since mocking occurs at the Spring MVC layer, code that relies on
lower-level servlet container behavior cannot be directly tested with MockMvc.
TIP For example, Spring Boot’s error handling is based on the “error page” support
provided by the servlet container. This means that, whilst you can test your MVC layer
throws and handles exceptions as expected, you cannot directly test that a specific
custom error page is rendered. If you need to test these lower-level concerns, you can
start a fully running server as described in the next section.
If you need to start a full running server, we recommend that you use random ports. If you use
@SpringBootTest(webEnvironment=WebEnvironment.RANDOM_PORT), an available port is picked at random
each time your test runs.
The @LocalServerPort annotation can be used to inject the actual port used into your test. For
convenience, tests that need to make REST calls to the started server can additionally @Autowire a
WebTestClient, which resolves relative links to the running server and comes with a dedicated API
for verifying responses, as shown in the following example:
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.test.web.reactive.server.WebTestClient;
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortWebTestClientTests {
@Test
void exampleTest(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
150
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment
import org.springframework.test.web.reactive.server.WebTestClient
import org.springframework.test.web.reactive.server.expectBody
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortWebTestClientTests {
@Test
fun exampleTest(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
TIP WebTestClient can be used against both live servers and mock environments.
This setup requires spring-webflux on the classpath. If you can not or will not add webflux, Spring
Boot also provides a TestRestTemplate facility:
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.boot.test.web.client.TestRestTemplate;
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortTestRestTemplateTests {
@Test
void exampleTest(@Autowired TestRestTemplate restTemplate) {
String body = restTemplate.getForObject("/", String.class);
assertThat(body).isEqualTo("Hello World");
}
151
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment
import org.springframework.boot.test.web.client.TestRestTemplate
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortTestRestTemplateTests {
@Test
fun exampleTest(@Autowired restTemplate: TestRestTemplate) {
val body = restTemplate.getForObject("/", String::class.java)
assertThat(body).isEqualTo("Hello World")
}
Customizing WebTestClient
Using JMX
As the test context framework caches context, JMX is disabled by default to prevent identical
components to register on the same domain. If such test needs access to an MBeanServer, consider
marking it dirty as well:
152
Java
import javax.management.MBeanServer;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.annotation.DirtiesContext;
@SpringBootTest(properties = "spring.jmx.enabled=true")
@DirtiesContext
class MyJmxTests {
@Autowired
private MBeanServer mBeanServer;
@Test
void exampleTest() {
assertThat(this.mBeanServer.getDomains()).contains("java.lang");
// ...
}
Kotlin
import javax.management.MBeanServer
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.test.annotation.DirtiesContext
@SpringBootTest(properties = ["spring.jmx.enabled=true"])
@DirtiesContext
class MyJmxTests(@Autowired val mBeanServer: MBeanServer) {
@Test
fun exampleTest() {
assertThat(mBeanServer.domains).contains("java.lang")
// ...
}
153
Using Metrics
Regardless of your classpath, meter registries, except the in-memory backed, are not auto-
configured when using @SpringBootTest.
If you need to export metrics to a different backend as part of an integration test, annotate it with
@AutoConfigureObservability.
Using Tracing
When running tests, it is sometimes necessary to mock certain components within your application
context. For example, you may have a facade over some remote service that is unavailable during
development. Mocking can also be useful when you want to simulate failures that might be hard to
trigger in a real environment.
Spring Boot includes a @MockBean annotation that can be used to define a Mockito mock for a bean
inside your ApplicationContext. You can use the annotation to add new beans or replace a single
existing bean definition. The annotation can be used directly on test classes, on fields within your
test, or on @Configuration classes and fields. When used on a field, the instance of the created mock
is also injected. Mock beans are automatically reset after each test method.
154
If your test uses one of Spring Boot’s test annotations (such as @SpringBootTest), this
feature is automatically enabled. To use this feature with a different arrangement,
listeners must be explicitly added, as shown in the following example:
Java
import
org.springframework.boot.test.mock.mockito.MockitoTestExecutionListener;
import
org.springframework.boot.test.mock.mockito.ResetMocksTestExecutionListen
er;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.TestExecutionListeners;
@ContextConfiguration(classes = MyConfig.class)
@TestExecutionListeners({ MockitoTestExecutionListener.class,
ResetMocksTestExecutionListener.class })
class MyTests {
// ...
}
NOTE
Kotlin
import
org.springframework.boot.test.mock.mockito.MockitoTestExecutionListener
import
org.springframework.boot.test.mock.mockito.ResetMocksTestExecutionListen
er
import org.springframework.test.context.ContextConfiguration
import org.springframework.test.context.TestExecutionListeners
@ContextConfiguration(classes = [MyConfig::class])
@TestExecutionListeners(
MockitoTestExecutionListener::class,
ResetMocksTestExecutionListener::class
)
class MyTests {
// ...
The following example replaces an existing RemoteService bean with a mock implementation:
155
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.mock.mockito.MockBean;
@SpringBootTest
class MyTests {
@Autowired
private Reverser reverser;
@MockBean
private RemoteService remoteService;
@Test
void exampleTest() {
given(this.remoteService.getValue()).willReturn("spring");
String reverse = this.reverser.getReverseValue(); // Calls injected
RemoteService
assertThat(reverse).isEqualTo("gnirps");
}
156
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.mockito.BDDMockito.given
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.mock.mockito.MockBean
@SpringBootTest
class MyTests(@Autowired val reverser: Reverser, @MockBean val remoteService:
RemoteService) {
@Test
fun exampleTest() {
given(remoteService.value).willReturn("spring")
val reverse = reverser.reverseValue // Calls injected RemoteService
assertThat(reverse).isEqualTo("gnirps")
}
@MockBean cannot be used to mock the behavior of a bean that is exercised during
application context refresh. By the time the test is executed, the application context
NOTE
refresh has completed and it is too late to configure the mocked behavior. We
recommend using a @Bean method to create and configure the mock in this situation.
Additionally, you can use @SpyBean to wrap any existing bean with a Mockito spy. See the Javadoc for
full details.
CGLib proxies, such as those created for scoped beans, declare the proxied methods
as final. This stops Mockito from functioning correctly as it cannot mock or spy on
final methods in its default configuration. If you want to mock or spy on such a
NOTE
bean, configure Mockito to use its inline mock maker by adding
org.mockito:mockito-inline to your application’s test dependencies. This allows
Mockito to mock and spy on final methods.
While Spring’s test framework caches application contexts between tests and reuses
NOTE a context for tests sharing the same configuration, the use of @MockBean or @SpyBean
influences the cache key, which will most likely increase the number of contexts.
If you are using @SpyBean to spy on a bean with @Cacheable methods that refer to
parameters by name, your application must be compiled with -parameters. This
TIP
ensures that the parameter names are available to the caching infrastructure once the
bean has been spied upon.
157
When you are using @SpyBean to spy on a bean that is proxied by Spring, you may need
TIP to remove Spring’s proxy in some situations, for example when setting expectations
using given or when. Use AopTestUtils.getTargetObject(yourProxiedSpy) to do so.
Auto-configured Tests
Spring Boot’s auto-configuration system works well for applications but can sometimes be a little
too much for tests. It often helps to load only the parts of the configuration that are required to test
a “slice” of your application. For example, you might want to test that Spring MVC controllers are
mapping URLs correctly, and you do not want to involve database calls in those tests, or you might
want to test JPA entities, and you are not interested in the web layer when those tests run.
Each slice restricts component scan to appropriate components and loads a very
restricted set of auto-configuration classes. If you need to exclude one of them, most
NOTE
@…Test annotations provide an excludeAutoConfiguration attribute. Alternatively,
you can use @ImportAutoConfiguration#exclude.
Including multiple “slices” by using several @…Test annotations in one test is not
NOTE supported. If you need multiple “slices”, pick one of the @…Test annotations and
include the @AutoConfigure… annotations of the other “slices” by hand.
To test that object JSON serialization and deserialization is working as expected, you can use the
@JsonTest annotation. @JsonTest auto-configures the available supported JSON mapper, which can
be one of the following libraries:
• Gson
• Jsonb
A list of the auto-configurations that are enabled by @JsonTest can be found in the
TIP
appendix.
If you need to configure elements of the auto-configuration, you can use the
@AutoConfigureJsonTesters annotation.
Spring Boot includes AssertJ-based helpers that work with the JSONAssert and JsonPath libraries to
158
check that JSON appears as expected. The JacksonTester, GsonTester, JsonbTester, and
BasicJsonTester classes can be used for Jackson, Gson, Jsonb, and Strings respectively. Any helper
fields on the test class can be @Autowired when using @JsonTest. The following example shows a test
class for Jackson:
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.json.JsonTest;
import org.springframework.boot.test.json.JacksonTester;
@JsonTest
class MyJsonTests {
@Autowired
private JacksonTester<VehicleDetails> json;
@Test
void serialize() throws Exception {
VehicleDetails details = new VehicleDetails("Honda", "Civic");
// Assert against a `.json` file in the same package as the test
assertThat(this.json.write(details)).isEqualToJson("expected.json");
// Or use JSON path based assertions
assertThat(this.json.write(details)).hasJsonPathStringValue("@.make");
assertThat(this.json.write(details)).extractingJsonPathStringValue("@.make").isEqualTo
("Honda");
}
@Test
void deserialize() throws Exception {
String content = "{\"make\":\"Ford\",\"model\":\"Focus\"}";
assertThat(this.json.parse(content)).isEqualTo(new VehicleDetails("Ford",
"Focus"));
assertThat(this.json.parseObject(content).getMake()).isEqualTo("Ford");
}
159
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.json.JsonTest
import org.springframework.boot.test.json.JacksonTester
@JsonTest
class MyJsonTests(@Autowired val json: JacksonTester<VehicleDetails>) {
@Test
fun serialize() {
val details = VehicleDetails("Honda", "Civic")
// Assert against a `.json` file in the same package as the test
assertThat(json.write(details)).isEqualToJson("expected.json")
// Or use JSON path based assertions
assertThat(json.write(details)).hasJsonPathStringValue("@.make")
assertThat(json.write(details)).extractingJsonPathStringValue("@.make").isEqualTo("Hon
da")
}
@Test
fun deserialize() {
val content = "{\"make\":\"Ford\",\"model\":\"Focus\"}"
assertThat(json.parse(content)).isEqualTo(VehicleDetails("Ford", "Focus"))
assertThat(json.parseObject(content).make).isEqualTo("Ford")
}
JSON helper classes can also be used directly in standard unit tests. To do so, call the
NOTE
initFields method of the helper in your @Before method if you do not use @JsonTest.
If you use Spring Boot’s AssertJ-based helpers to assert on a number value at a given JSON path, you
might not be able to use isEqualTo depending on the type. Instead, you can use AssertJ’s satisfies to
assert that the value matches the given condition. For instance, the following example asserts that
the actual number is a float value close to 0.15 within an offset of 0.01.
160
Java
@Test
void someTest() throws Exception {
SomeObject value = new SomeObject(0.152f);
assertThat(this.json.write(value)).extractingJsonPathNumberValue("@.test.numberValue")
.satisfies((number) -> assertThat(number.floatValue()).isCloseTo(0.15f,
within(0.01f)));
}
Kotlin
@Test
fun someTest() {
val value = SomeObject(0.152f)
assertThat(json.write(value)).extractingJsonPathNumberValue("@.test.numberValue")
.satisfies(ThrowingConsumer { number ->
assertThat(number.toFloat()).isCloseTo(0.15f, within(0.01f))
})
}
To test whether Spring MVC controllers are working as expected, use the @WebMvcTest annotation.
@WebMvcTest auto-configures the Spring MVC infrastructure and limits scanned beans to @Controller,
@ControllerAdvice, @JsonComponent, Converter, GenericConverter, Filter, HandlerInterceptor,
WebMvcConfigurer, WebMvcRegistrations, and HandlerMethodArgumentResolver. Regular @Component and
@ConfigurationProperties beans are not scanned when the @WebMvcTest annotation is used.
@EnableConfigurationProperties can be used to include @ConfigurationProperties beans.
A list of the auto-configuration settings that are enabled by @WebMvcTest can be found
TIP
in the appendix.
If you need to register extra components, such as the Jackson Module, you can import
TIP
additional configuration classes by using @Import on your test.
Often, @WebMvcTest is limited to a single controller and is used in combination with @MockBean to
provide mock implementations for required collaborators.
@WebMvcTest also auto-configures MockMvc. Mock MVC offers a powerful way to quickly test MVC
controllers without needing to start a full HTTP server.
161
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.boot.test.mock.mockito.MockBean;
import org.springframework.http.MediaType;
import org.springframework.test.web.servlet.MockMvc;
@WebMvcTest(UserVehicleController.class)
class MyControllerTests {
@Autowired
private MockMvc mvc;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() throws Exception {
given(this.userVehicleService.getVehicleDetails("sboot"))
.willReturn(new VehicleDetails("Honda", "Civic"));
this.mvc.perform(get("/sboot/vehicle").accept(MediaType.TEXT_PLAIN))
.andExpect(status().isOk())
.andExpect(content().string("Honda Civic"));
}
162
Kotlin
import org.junit.jupiter.api.Test
import org.mockito.BDDMockito.given
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
import org.springframework.boot.test.mock.mockito.MockBean
import org.springframework.http.MediaType
import org.springframework.test.web.servlet.MockMvc
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders
import org.springframework.test.web.servlet.result.MockMvcResultMatchers
@WebMvcTest(UserVehicleController::class)
class MyControllerTests(@Autowired val mvc: MockMvc) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot"))
.willReturn(VehicleDetails("Honda", "Civic"))
mvc.perform(MockMvcRequestBuilders.get("/sboot/vehicle").accept(MediaType.TEXT_PLAIN))
.andExpect(MockMvcResultMatchers.status().isOk)
.andExpect(MockMvcResultMatchers.content().string("Honda Civic"))
}
If you need to configure elements of the auto-configuration (for example, when servlet
TIP filters should be applied) you can use attributes in the @AutoConfigureMockMvc
annotation.
If you use HtmlUnit and Selenium, auto-configuration also provides an HtmlUnit WebClient bean
and/or a Selenium WebDriver bean. The following example uses HtmlUnit:
163
Java
import com.gargoylesoftware.htmlunit.WebClient;
import com.gargoylesoftware.htmlunit.html.HtmlPage;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.boot.test.mock.mockito.MockBean;
@WebMvcTest(UserVehicleController.class)
class MyHtmlUnitTests {
@Autowired
private WebClient webClient;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() throws Exception {
given(this.userVehicleService.getVehicleDetails("sboot")).willReturn(new
VehicleDetails("Honda", "Civic"));
HtmlPage page = this.webClient.getPage("/sboot/vehicle.html");
assertThat(page.getBody().getTextContent()).isEqualTo("Honda Civic");
}
164
Kotlin
import com.gargoylesoftware.htmlunit.WebClient
import com.gargoylesoftware.htmlunit.html.HtmlPage
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.mockito.BDDMockito.given
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
import org.springframework.boot.test.mock.mockito.MockBean
@WebMvcTest(UserVehicleController::class)
class MyHtmlUnitTests(@Autowired val webClient: WebClient) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot")).willReturn(VehicleDetails("Honda"
, "Civic"))
val page = webClient.getPage<HtmlPage>("/sboot/vehicle.html")
assertThat(page.body.textContent).isEqualTo("Honda Civic")
}
By default, Spring Boot puts WebDriver beans in a special “scope” to ensure that the
NOTE driver exits after each test and that a new instance is injected. If you do not want
this behavior, you can add @Scope("singleton") to your WebDriver @Bean definition.
The webDriver scope created by Spring Boot will replace any user defined scope
WARNING of the same name. If you define your own webDriver scope you may find it stops
working when you use @WebMvcTest.
If you have Spring Security on the classpath, @WebMvcTest will also scan WebSecurityConfigurer
beans. Instead of disabling security completely for such tests, you can use Spring Security’s test
support. More details on how to use Spring Security’s MockMvc support can be found in this Testing
With Spring Security how-to section.
Sometimes writing Spring MVC tests is not enough; Spring Boot can help you run full
TIP
end-to-end tests with an actual server.
To test that Spring WebFlux controllers are working as expected, you can use the @WebFluxTest
annotation. @WebFluxTest auto-configures the Spring WebFlux infrastructure and limits scanned
165
beans to @Controller, @ControllerAdvice, @JsonComponent, Converter, GenericConverter, WebFilter, and
WebFluxConfigurer. Regular @Component and @ConfigurationProperties beans are not scanned when
the @WebFluxTest annotation is used. @EnableConfigurationProperties can be used to include
@ConfigurationProperties beans.
A list of the auto-configurations that are enabled by @WebFluxTest can be found in the
TIP
appendix.
If you need to register extra components, such as Jackson Module, you can import
TIP
additional configuration classes using @Import on your test.
Often, @WebFluxTest is limited to a single controller and used in combination with the @MockBean
annotation to provide mock implementations for required collaborators.
@WebFluxTest also auto-configures WebTestClient, which offers a powerful way to quickly test
WebFlux controllers without needing to start a full HTTP server.
166
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest;
import org.springframework.boot.test.mock.mockito.MockBean;
import org.springframework.http.MediaType;
import org.springframework.test.web.reactive.server.WebTestClient;
@WebFluxTest(UserVehicleController.class)
class MyControllerTests {
@Autowired
private WebTestClient webClient;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() {
given(this.userVehicleService.getVehicleDetails("sboot"))
.willReturn(new VehicleDetails("Honda", "Civic"));
this.webClient.get().uri("/sboot/vehicle").accept(MediaType.TEXT_PLAIN).exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Honda Civic");
}
167
Kotlin
import org.junit.jupiter.api.Test
import org.mockito.BDDMockito.given
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest
import org.springframework.boot.test.mock.mockito.MockBean
import org.springframework.http.MediaType
import org.springframework.test.web.reactive.server.WebTestClient
import org.springframework.test.web.reactive.server.expectBody
@WebFluxTest(UserVehicleController::class)
class MyControllerTests(@Autowired val webClient: WebTestClient) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot"))
.willReturn(VehicleDetails("Honda", "Civic"))
webClient.get().uri("/sboot/vehicle").accept(MediaType.TEXT_PLAIN).exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Honda Civic")
}
Sometimes writing Spring WebFlux tests is not enough; Spring Boot can help you run
TIP
full end-to-end tests with an actual server.
Spring GraphQL offers a dedicated testing support module; you’ll need to add it to your project:
168
Maven
<dependencies>
<dependency>
<groupId>org.springframework.graphql</groupId>
<artifactId>spring-graphql-test</artifactId>
<scope>test</scope>
</dependency>
<!-- Unless already present in the compile scope -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-webflux</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
Gradle
dependencies {
testImplementation("org.springframework.graphql:spring-graphql-test")
// Unless already present in the implementation configuration
testImplementation("org.springframework.boot:spring-boot-starter-webflux")
}
This testing module ships the GraphQlTester. The tester is heavily used in test, so be sure to become
familiar with using it. There are GraphQlTester variants and Spring Boot will auto-configure them
depending on the type of tests:
• the ExecutionGraphQlServiceTester performs tests on the server side, without a client nor a
transport
• the HttpGraphQlTester performs tests with a client that connects to a server, with or without a
live server
Spring Boot helps you to test your Spring GraphQL Controllers with the @GraphQlTest annotation.
@GraphQlTest auto-configures the Spring GraphQL infrastructure, without any transport nor server
being involved. This limits scanned beans to @Controller, RuntimeWiringConfigurer, JsonComponent,
Converter, GenericConverter, DataFetcherExceptionResolver, Instrumentation and
GraphQlSourceBuilderCustomizer. Regular @Component and @ConfigurationProperties beans are not
scanned when the @GraphQlTest annotation is used. @EnableConfigurationProperties can be used to
include @ConfigurationProperties beans.
A list of the auto-configurations that are enabled by @GraphQlTest can be found in the
TIP
appendix.
Often, @GraphQlTest is limited to a set of controllers and used in combination with the @MockBean
annotation to provide mock implementations for required collaborators.
169
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.docs.web.graphql.runtimewiring.GreetingController;
import org.springframework.boot.test.autoconfigure.graphql.GraphQlTest;
import org.springframework.graphql.test.tester.GraphQlTester;
@GraphQlTest(GreetingController.class)
class GreetingControllerTests {
@Autowired
private GraphQlTester graphQlTester;
@Test
void shouldGreetWithSpecificName() {
this.graphQlTester.document("{ greeting(name: \"Alice\") } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Alice!");
}
@Test
void shouldGreetWithDefaultName() {
this.graphQlTester.document("{ greeting } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Spring!");
}
170
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.docs.web.graphql.runtimewiring.GreetingController
import org.springframework.boot.test.autoconfigure.graphql.GraphQlTest
import org.springframework.graphql.test.tester.GraphQlTester
@GraphQlTest(GreetingController::class)
internal class GreetingControllerTests {
@Autowired
lateinit var graphQlTester: GraphQlTester
@Test
fun shouldGreetWithSpecificName() {
graphQlTester.document("{ greeting(name: \"Alice\") }
").execute().path("greeting").entity(String::class.java)
.isEqualTo("Hello, Alice!")
}
@Test
fun shouldGreetWithDefaultName() {
graphQlTester.document("{ greeting }
").execute().path("greeting").entity(String::class.java)
.isEqualTo("Hello, Spring!")
}
@SpringBootTest tests are full integration tests and involve the entire application. When using a
random or defined port, a live server is configured and an HttpGraphQlTester bean is contributed
automatically so you can use it to test your server. When a MOCK environment is configured, you
can also request an HttpGraphQlTester bean by annotating your test class with
@AutoConfigureHttpGraphQlTester:
171
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import
org.springframework.boot.test.autoconfigure.graphql.tester.AutoConfigureHttpGraphQlTes
ter;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.graphql.test.tester.HttpGraphQlTester;
@AutoConfigureHttpGraphQlTester
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
class GraphQlIntegrationTests {
@Test
void shouldGreetWithSpecificName(@Autowired HttpGraphQlTester graphQlTester) {
HttpGraphQlTester authenticatedTester = graphQlTester.mutate()
.webTestClient((client) -> client.defaultHeaders((headers) ->
headers.setBasicAuth("admin", "ilovespring")))
.build();
authenticatedTester.document("{ greeting(name: \"Alice\") } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Alice!");
}
172
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import
org.springframework.boot.test.autoconfigure.graphql.tester.AutoConfigureHttpGraphQlTes
ter
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.graphql.test.tester.HttpGraphQlTester
import org.springframework.http.HttpHeaders
import org.springframework.test.web.reactive.server.WebTestClient
@AutoConfigureHttpGraphQlTester
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
class GraphQlIntegrationTests {
@Test
fun shouldGreetWithSpecificName(@Autowired graphQlTester: HttpGraphQlTester) {
val authenticatedTester = graphQlTester.mutate()
.webTestClient { client: WebTestClient.Builder ->
client.defaultHeaders { headers: HttpHeaders ->
headers.setBasicAuth("admin", "ilovespring")
}
}.build()
authenticatedTester.document("{ greeting(name: \"Alice\") } ").execute()
.path("greeting").entity(String::class.java).isEqualTo("Hello, Alice!")
}
}
The following example shows a typical setup for using Cassandra tests in Spring Boot:
173
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.cassandra.DataCassandraTest;
@DataCassandraTest
class MyDataCassandraTests {
@Autowired
private SomeRepository repository;
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.cassandra.DataCassandraTest
@DataCassandraTest
class MyDataCassandraTests(@Autowired val repository: SomeRepository)
The following example shows a typical setup for using Couchbase tests in Spring Boot:
174
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.couchbase.DataCouchbaseTest;
@DataCouchbaseTest
class MyDataCouchbaseTests {
@Autowired
private SomeRepository repository;
// ...
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.couchbase.DataCouchbaseTest
@DataCouchbaseTest
class MyDataCouchbaseTests(@Autowired val repository: SomeRepository) {
// ...
The following example shows a typical setup for using Elasticsearch tests in Spring Boot:
175
Java
import org.springframework.beans.factory.annotation.Autowired;
import
org.springframework.boot.test.autoconfigure.data.elasticsearch.DataElasticsearchTest;
@DataElasticsearchTest
class MyDataElasticsearchTests {
@Autowired
private SomeRepository repository;
// ...
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import
org.springframework.boot.test.autoconfigure.data.elasticsearch.DataElasticsearchTest
@DataElasticsearchTest
class MyDataElasticsearchTests(@Autowired val repository: SomeRepository) {
// ...
You can use the @DataJpaTest annotation to test JPA applications. By default, it scans for @Entity
classes and configures Spring Data JPA repositories. If an embedded database is available on the
classpath, it configures one as well. SQL queries are logged by default by setting the
spring.jpa.show-sql property to true. This can be disabled using the showSql attribute of the
annotation.
Regular @Component and @ConfigurationProperties beans are not scanned when the @DataJpaTest
annotation is used. @EnableConfigurationProperties can be used to include @ConfigurationProperties
beans.
A list of the auto-configuration settings that are enabled by @DataJpaTest can be found
TIP
in the appendix.
By default, data JPA tests are transactional and roll back at the end of each test. See the relevant
section in the Spring Framework Reference Documentation for more details. If that is not what you
want, you can disable transaction management for a test or for the whole class as follows:
176
Java
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@DataJpaTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyNonTransactionalTests {
// ...
Kotlin
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest
import org.springframework.transaction.annotation.Propagation
import org.springframework.transaction.annotation.Transactional
@DataJpaTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyNonTransactionalTests {
// ...
Data JPA tests may also inject a TestEntityManager bean, which provides an alternative to the
standard JPA EntityManager that is specifically designed for tests.
A JdbcTemplate is also available if you need that. The following example shows the @DataJpaTest
annotation in use:
177
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
import org.springframework.boot.test.autoconfigure.orm.jpa.TestEntityManager;
@DataJpaTest
class MyRepositoryTests {
@Autowired
private TestEntityManager entityManager;
@Autowired
private UserRepository repository;
@Test
void testExample() {
this.entityManager.persist(new User("sboot", "1234"));
User user = this.repository.findByUsername("sboot");
assertThat(user.getUsername()).isEqualTo("sboot");
assertThat(user.getEmployeeNumber()).isEqualTo("1234");
}
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest
import org.springframework.boot.test.autoconfigure.orm.jpa.TestEntityManager
@DataJpaTest
class MyRepositoryTests(@Autowired val entityManager: TestEntityManager, @Autowired
val repository: UserRepository) {
@Test
fun testExample() {
entityManager.persist(User("sboot", "1234"))
val user = repository.findByUsername("sboot")
assertThat(user?.username).isEqualTo("sboot")
assertThat(user?.employeeNumber).isEqualTo("1234")
}
178
In-memory embedded databases generally work well for tests, since they are fast and do not
require any installation. If, however, you prefer to run tests against a real database you can use the
@AutoConfigureTestDatabase annotation, as shown in the following example:
Java
import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
import
org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase.Replace;
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
@DataJpaTest
@AutoConfigureTestDatabase(replace = Replace.NONE)
class MyRepositoryTests {
// ...
Kotlin
import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest
@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
class MyRepositoryTests {
// ...
@JdbcTest is similar to @DataJpaTest but is for tests that only require a DataSource and do not use
Spring Data JDBC. By default, it configures an in-memory embedded database and a JdbcTemplate.
Regular @Component and @ConfigurationProperties beans are not scanned when the @JdbcTest
annotation is used. @EnableConfigurationProperties can be used to include @ConfigurationProperties
beans.
A list of the auto-configurations that are enabled by @JdbcTest can be found in the
TIP
appendix.
By default, JDBC tests are transactional and roll back at the end of each test. See the relevant section
in the Spring Framework Reference Documentation for more details. If that is not what you want,
you can disable transaction management for a test or for the whole class, as follows:
179
Java
import org.springframework.boot.test.autoconfigure.jdbc.JdbcTest;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@JdbcTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyTransactionalTests {
Kotlin
import org.springframework.boot.test.autoconfigure.jdbc.JdbcTest
import org.springframework.transaction.annotation.Propagation
import org.springframework.transaction.annotation.Transactional
@JdbcTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyTransactionalTests
If you prefer your test to run against a real database, you can use the @AutoConfigureTestDatabase
annotation in the same way as for DataJpaTest. (See "Auto-configured Data JPA Tests".)
@DataJdbcTest is similar to @JdbcTest but is for tests that use Spring Data JDBC repositories. By
default, it configures an in-memory embedded database, a JdbcTemplate, and Spring Data JDBC
repositories. Only AbstractJdbcConfiguration subclasses are scanned when the @DataJdbcTest
annotation is used, regular @Component and @ConfigurationProperties beans are not scanned.
@EnableConfigurationProperties can be used to include @ConfigurationProperties beans.
A list of the auto-configurations that are enabled by @DataJdbcTest can be found in the
TIP
appendix.
By default, Data JDBC tests are transactional and roll back at the end of each test. See the relevant
section in the Spring Framework Reference Documentation for more details. If that is not what you
want, you can disable transaction management for a test or for the whole test class as shown in the
JDBC example.
If you prefer your test to run against a real database, you can use the @AutoConfigureTestDatabase
annotation in the same way as for DataJpaTest. (See "Auto-configured Data JPA Tests".)
You can use @JooqTest in a similar fashion as @JdbcTest but for jOOQ-related tests. As jOOQ relies
heavily on a Java-based schema that corresponds with the database schema, the existing DataSource
is used. If you want to replace it with an in-memory database, you can use
180
@AutoConfigureTestDatabase to override those settings. (For more about using jOOQ with Spring
Boot, see "Using jOOQ".) Regular @Component and @ConfigurationProperties beans are not scanned
when the @JooqTest annotation is used. @EnableConfigurationProperties can be used to include
@ConfigurationProperties beans.
A list of the auto-configurations that are enabled by @JooqTest can be found in the
TIP
appendix.
@JooqTest configures a DSLContext. The following example shows the @JooqTest annotation in use:
Java
import org.jooq.DSLContext;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.jooq.JooqTest;
@JooqTest
class MyJooqTests {
@Autowired
private DSLContext dslContext;
// ...
Kotlin
import org.jooq.DSLContext
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.jooq.JooqTest
@JooqTest
class MyJooqTests(@Autowired val dslContext: DSLContext) {
// ...
JOOQ tests are transactional and roll back at the end of each test by default. If that is not what you
want, you can disable transaction management for a test or for the whole test class as shown in the
JDBC example.
You can use @DataMongoTest to test MongoDB applications. By default, it configures a MongoTemplate,
scans for @Document classes, and configures Spring Data MongoDB repositories. Regular @Component
and @ConfigurationProperties beans are not scanned when the @DataMongoTest annotation is used.
181
@EnableConfigurationProperties can be used to include @ConfigurationProperties beans. (For more
about using MongoDB with Spring Boot, see "MongoDB".)
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.mongo.DataMongoTest;
import org.springframework.data.mongodb.core.MongoTemplate;
@DataMongoTest
class MyDataMongoDbTests {
@Autowired
private MongoTemplate mongoTemplate;
// ...
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.mongo.DataMongoTest
import org.springframework.data.mongodb.core.MongoTemplate
@DataMongoTest
class MyDataMongoDbTests(@Autowired val mongoTemplate: MongoTemplate) {
// ...
You can use @DataNeo4jTest to test Neo4j applications. By default, it scans for @Node classes, and
configures Spring Data Neo4j repositories. Regular @Component and @ConfigurationProperties beans
are not scanned when the @DataNeo4jTest annotation is used. @EnableConfigurationProperties can be
used to include @ConfigurationProperties beans. (For more about using Neo4J with Spring Boot, see
"Neo4j".)
The following example shows a typical setup for using Neo4J tests in Spring Boot:
182
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.neo4j.DataNeo4jTest;
@DataNeo4jTest
class MyDataNeo4jTests {
@Autowired
private SomeRepository repository;
// ...
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.neo4j.DataNeo4jTest
@DataNeo4jTest
class MyDataNeo4jTests(@Autowired val repository: SomeRepository) {
// ...
By default, Data Neo4j tests are transactional and roll back at the end of each test. See the relevant
section in the Spring Framework Reference Documentation for more details. If that is not what you
want, you can disable transaction management for a test or for the whole class, as follows:
Java
import org.springframework.boot.test.autoconfigure.data.neo4j.DataNeo4jTest;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@DataNeo4jTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyDataNeo4jTests {
183
Kotlin
import org.springframework.boot.test.autoconfigure.data.neo4j.DataNeo4jTest
import org.springframework.transaction.annotation.Propagation
import org.springframework.transaction.annotation.Transactional
@DataNeo4jTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyDataNeo4jTests
Transactional tests are not supported with reactive access. If you are using this
NOTE
style, you must configure @DataNeo4jTest tests as described above.
You can use @DataRedisTest to test Redis applications. By default, it scans for @RedisHash classes and
configures Spring Data Redis repositories. Regular @Component and @ConfigurationProperties beans
are not scanned when the @DataRedisTest annotation is used. @EnableConfigurationProperties can be
used to include @ConfigurationProperties beans. (For more about using Redis with Spring Boot, see
"Redis".)
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.redis.DataRedisTest;
@DataRedisTest
class MyDataRedisTests {
@Autowired
private SomeRepository repository;
// ...
184
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.redis.DataRedisTest
@DataRedisTest
class MyDataRedisTests(@Autowired val repository: SomeRepository) {
// ...
You can use @DataLdapTest to test LDAP applications. By default, it configures an in-memory
embedded LDAP (if available), configures an LdapTemplate, scans for @Entry classes, and configures
Spring Data LDAP repositories. Regular @Component and @ConfigurationProperties beans are not
scanned when the @DataLdapTest annotation is used. @EnableConfigurationProperties can be used to
include @ConfigurationProperties beans. (For more about using LDAP with Spring Boot, see "LDAP".)
A list of the auto-configuration settings that are enabled by @DataLdapTest can be found
TIP
in the appendix.
Java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.data.ldap.DataLdapTest;
import org.springframework.ldap.core.LdapTemplate;
@DataLdapTest
class MyDataLdapTests {
@Autowired
private LdapTemplate ldapTemplate;
// ...
185
Kotlin
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.data.ldap.DataLdapTest
import org.springframework.ldap.core.LdapTemplate
@DataLdapTest
class MyDataLdapTests(@Autowired val ldapTemplate: LdapTemplate) {
// ...
In-memory embedded LDAP generally works well for tests, since it is fast and does not require any
developer installation. If, however, you prefer to run tests against a real LDAP server, you should
exclude the embedded LDAP auto-configuration, as shown in the following example:
Java
import
org.springframework.boot.autoconfigure.ldap.embedded.EmbeddedLdapAutoConfiguration;
import org.springframework.boot.test.autoconfigure.data.ldap.DataLdapTest;
@DataLdapTest(excludeAutoConfiguration = EmbeddedLdapAutoConfiguration.class)
class MyDataLdapTests {
// ...
Kotlin
import
org.springframework.boot.autoconfigure.ldap.embedded.EmbeddedLdapAutoConfiguration
import org.springframework.boot.test.autoconfigure.data.ldap.DataLdapTest
@DataLdapTest(excludeAutoConfiguration = [EmbeddedLdapAutoConfiguration::class])
class MyDataLdapTests {
// ...
You can use the @RestClientTest annotation to test REST clients. By default, it auto-configures
Jackson, GSON, and Jsonb support, configures a RestTemplateBuilder, and adds support for
MockRestServiceServer. Regular @Component and @ConfigurationProperties beans are not scanned
when the @RestClientTest annotation is used. @EnableConfigurationProperties can be used to
186
include @ConfigurationProperties beans.
The specific beans that you want to test should be specified by using the value or components
attribute of @RestClientTest, as shown in the following example:
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.client.RestClientTest;
import org.springframework.http.MediaType;
import org.springframework.test.web.client.MockRestServiceServer;
@RestClientTest(RemoteVehicleDetailsService.class)
class MyRestClientTests {
@Autowired
private RemoteVehicleDetailsService service;
@Autowired
private MockRestServiceServer server;
@Test
void getVehicleDetailsWhenResultIsSuccessShouldReturnDetails() {
this.server.expect(requestTo("/greet/details")).andRespond(withSuccess("hello",
MediaType.TEXT_PLAIN));
String greeting = this.service.callRestService();
assertThat(greeting).isEqualTo("hello");
}
187
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.client.RestClientTest
import org.springframework.http.MediaType
import org.springframework.test.web.client.MockRestServiceServer
import org.springframework.test.web.client.match.MockRestRequestMatchers
import org.springframework.test.web.client.response.MockRestResponseCreators
@RestClientTest(RemoteVehicleDetailsService::class)
class MyRestClientTests(
@Autowired val service: RemoteVehicleDetailsService,
@Autowired val server: MockRestServiceServer) {
@Test
fun getVehicleDetailsWhenResultIsSuccessShouldReturnDetails(): Unit {
server.expect(MockRestRequestMatchers.requestTo("/greet/details"))
.andRespond(MockRestResponseCreators.withSuccess("hello",
MediaType.TEXT_PLAIN))
val greeting = service.callRestService()
assertThat(greeting).isEqualTo("hello")
}
You can use the @AutoConfigureRestDocs annotation to use Spring REST Docs in your tests with Mock
MVC, REST Assured, or WebTestClient. It removes the need for the JUnit extension in Spring REST
Docs.
@AutoConfigureRestDocs customizes the MockMvc bean to use Spring REST Docs when testing servlet-
based web applications. You can inject it by using @Autowired and use it in your tests as you
normally would when using Mock MVC and Spring REST Docs, as shown in the following example:
188
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.http.MediaType;
import org.springframework.test.web.servlet.MockMvc;
@WebMvcTest(UserController.class)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Autowired
private MockMvc mvc;
@Test
void listUsers() throws Exception {
this.mvc.perform(get("/users").accept(MediaType.TEXT_PLAIN))
.andExpect(status().isOk())
.andDo(document("list-users"));
}
189
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
import org.springframework.http.MediaType
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentation
import org.springframework.test.web.servlet.MockMvc
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders
import org.springframework.test.web.servlet.result.MockMvcResultMatchers
@WebMvcTest(UserController::class)
@AutoConfigureRestDocs
class MyUserDocumentationTests(@Autowired val mvc: MockMvc) {
@Test
fun listUsers() {
mvc.perform(MockMvcRequestBuilders.get("/users").accept(MediaType.TEXT_PLAIN))
.andExpect(MockMvcResultMatchers.status().isOk)
.andDo(MockMvcRestDocumentation.document("list-users"))
}
If you require more control over Spring REST Docs configuration than offered by the attributes of
@AutoConfigureRestDocs, you can use a RestDocsMockMvcConfigurationCustomizer bean, as shown in
the following example:
Java
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsMockMvcConfigurationCusto
mizer;
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentationConfigurer;
import org.springframework.restdocs.templates.TemplateFormats;
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements RestDocsMockMvcConfigurationCustomizer
{
@Override
public void customize(MockMvcRestDocumentationConfigurer configurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown());
}
190
Kotlin
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsMockMvcConfigurationCusto
mizer
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentationConfigurer
import org.springframework.restdocs.templates.TemplateFormats
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsMockMvcConfigurationCustomizer {
If you want to make use of Spring REST Docs support for a parameterized output directory, you can
create a RestDocumentationResultHandler bean. The auto-configuration calls alwaysDo with this result
handler, thereby causing each MockMvc call to automatically generate the default snippets. The
following example shows a RestDocumentationResultHandler being defined:
Java
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentation;
import org.springframework.restdocs.mockmvc.RestDocumentationResultHandler;
@TestConfiguration(proxyBeanMethods = false)
public class MyResultHandlerConfiguration {
@Bean
public RestDocumentationResultHandler restDocumentation() {
return MockMvcRestDocumentation.document("{method-name}");
}
191
Kotlin
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.context.annotation.Bean
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentation
import org.springframework.restdocs.mockmvc.RestDocumentationResultHandler
@TestConfiguration(proxyBeanMethods = false)
class MyResultHandlerConfiguration {
@Bean
fun restDocumentation(): RestDocumentationResultHandler {
return MockMvcRestDocumentation.document("{method-name}")
}
@AutoConfigureRestDocs can also be used with WebTestClient when testing reactive web applications.
You can inject it by using @Autowired and use it in your tests as you normally would when using
@WebFluxTest and Spring REST Docs, as shown in the following example:
192
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs;
import org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest;
import org.springframework.test.web.reactive.server.WebTestClient;
import static
org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation.document;
@WebFluxTest
@AutoConfigureRestDocs
class MyUsersDocumentationTests {
@Autowired
private WebTestClient webTestClient;
@Test
void listUsers() {
this.webTestClient
.get().uri("/")
.exchange()
.expectStatus()
.isOk()
.expectBody()
.consumeWith(document("list-users"));
}
193
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs
import org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest
import org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation
import org.springframework.test.web.reactive.server.WebTestClient
@WebFluxTest
@AutoConfigureRestDocs
class MyUsersDocumentationTests(@Autowired val webTestClient: WebTestClient) {
@Test
fun listUsers() {
webTestClient
.get().uri("/")
.exchange()
.expectStatus()
.isOk
.expectBody()
.consumeWith(WebTestClientRestDocumentation.document("list-users"))
}
If you require more control over Spring REST Docs configuration than offered by the attributes of
@AutoConfigureRestDocs, you can use a RestDocsWebTestClientConfigurationCustomizer bean, as
shown in the following example:
Java
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsWebTestClientConfiguratio
nCustomizer;
import org.springframework.boot.test.context.TestConfiguration;
import
org.springframework.restdocs.webtestclient.WebTestClientRestDocumentationConfigurer;
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements
RestDocsWebTestClientConfigurationCustomizer {
@Override
public void customize(WebTestClientRestDocumentationConfigurer configurer) {
configurer.snippets().withEncoding("UTF-8");
}
194
Kotlin
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsWebTestClientConfiguratio
nCustomizer
import org.springframework.boot.test.context.TestConfiguration
import
org.springframework.restdocs.webtestclient.WebTestClientRestDocumentationConfigurer
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsWebTestClientConfigurationCustomizer {
If you want to make use of Spring REST Docs support for a parameterized output directory, you can
use a WebTestClientBuilderCustomizer to configure a consumer for every entity exchange result. The
following example shows such a WebTestClientBuilderCustomizer being defined:
Java
import org.springframework.boot.test.context.TestConfiguration;
import
org.springframework.boot.test.web.reactive.server.WebTestClientBuilderCustomizer;
import org.springframework.context.annotation.Bean;
import static
org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation.document;
@TestConfiguration(proxyBeanMethods = false)
public class MyWebTestClientBuilderCustomizerConfiguration {
@Bean
public WebTestClientBuilderCustomizer restDocumentation() {
return (builder) -> builder.entityExchangeResultConsumer(document("{method-
name}"));
}
195
Kotlin
import org.springframework.boot.test.context.TestConfiguration
import
org.springframework.boot.test.web.reactive.server.WebTestClientBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation
import org.springframework.test.web.reactive.server.WebTestClient
@TestConfiguration(proxyBeanMethods = false)
class MyWebTestClientBuilderCustomizerConfiguration {
@Bean
fun restDocumentation(): WebTestClientBuilderCustomizer {
return WebTestClientBuilderCustomizer { builder: WebTestClient.Builder ->
builder.entityExchangeResultConsumer(
WebTestClientRestDocumentation.document("{method-name}")
)
}
}
196
Java
import io.restassured.specification.RequestSpecification;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.boot.test.web.server.LocalServerPort;
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Test
void listUsers(@Autowired RequestSpecification documentationSpec, @LocalServerPort
int port) {
given(documentationSpec)
.filter(document("list-users"))
.when()
.port(port)
.get("/")
.then().assertThat()
.statusCode(is(200));
}
197
Kotlin
import io.restassured.RestAssured
import io.restassured.specification.RequestSpecification
import org.hamcrest.Matchers
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.restdocs.AutoConfigureRestDocs
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment
import org.springframework.boot.test.web.server.LocalServerPort
import org.springframework.restdocs.restassured.RestAssuredRestDocumentation
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Test
fun listUsers(@Autowired documentationSpec: RequestSpecification?,
@LocalServerPort port: Int) {
RestAssured.given(documentationSpec)
.filter(RestAssuredRestDocumentation.document("list-users"))
.`when`()
.port(port)["/"]
.then().assertThat()
.statusCode(Matchers.`is`(200))
}
If you require more control over Spring REST Docs configuration than offered by the attributes of
@AutoConfigureRestDocs, a RestDocsRestAssuredConfigurationCustomizer bean can be used, as shown
in the following example:
198
Java
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsRestAssuredConfigurationC
ustomizer;
import org.springframework.boot.test.context.TestConfiguration;
import
org.springframework.restdocs.restassured.RestAssuredRestDocumentationConfigurer;
import org.springframework.restdocs.templates.TemplateFormats;
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements
RestDocsRestAssuredConfigurationCustomizer {
@Override
public void customize(RestAssuredRestDocumentationConfigurer configurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown());
}
Kotlin
import
org.springframework.boot.test.autoconfigure.restdocs.RestDocsRestAssuredConfigurationC
ustomizer
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.restdocs.restassured.RestAssuredRestDocumentationConfigurer
import org.springframework.restdocs.templates.TemplateFormats
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsRestAssuredConfigurationCustomizer {
You can use @WebServiceClientTest to test applications that call web services using the Spring Web
Services project. By default, it configures a mock WebServiceServer bean and automatically
customizes your WebServiceTemplateBuilder. (For more about using Web Services with Spring Boot,
see "Web Services".)
199
A list of the auto-configuration settings that are enabled by @WebServiceClientTest can
TIP
be found in the appendix.
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import
org.springframework.boot.test.autoconfigure.webservices.client.WebServiceClientTest;
import org.springframework.ws.test.client.MockWebServiceServer;
import org.springframework.xml.transform.StringSource;
@WebServiceClientTest(SomeWebService.class)
class MyWebServiceClientTests {
@Autowired
private MockWebServiceServer server;
@Autowired
private SomeWebService someWebService;
@Test
void mockServerCall() {
this.server
.expect(payload(new StringSource("<request/>")))
.andRespond(withPayload(new
StringSource("<response><status>200</status></response>")));
assertThat(this.someWebService.test())
.extracting(Response::getStatus)
.isEqualTo(200);
}
200
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import
org.springframework.boot.test.autoconfigure.webservices.client.WebServiceClientTest
import org.springframework.ws.test.client.MockWebServiceServer
import org.springframework.ws.test.client.RequestMatchers
import org.springframework.ws.test.client.ResponseCreators
import org.springframework.xml.transform.StringSource
@WebServiceClientTest(SomeWebService::class)
class MyWebServiceClientTests(@Autowired val server: MockWebServiceServer, @Autowired
val someWebService: SomeWebService) {
@Test
fun mockServerCall() {
server
.expect(RequestMatchers.payload(StringSource("<request/>")))
.andRespond(ResponseCreators.withPayload(StringSource("<response><status>200</status><
/response>")))
assertThat(this.someWebService.test()).extracting(Response::status).isEqualTo(200)
}
You can use @WebServiceServerTest to test applications that implement web services using the
Spring Web Services project. By default, it configures a MockWebServiceClient bean that can be used
to call your web service endpoints. (For more about using Web Services with Spring Boot, see "Web
Services".)
201
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import
org.springframework.boot.test.autoconfigure.webservices.server.WebServiceServerTest;
import org.springframework.ws.test.server.MockWebServiceClient;
import org.springframework.ws.test.server.RequestCreators;
import org.springframework.ws.test.server.ResponseMatchers;
import org.springframework.xml.transform.StringSource;
@WebServiceServerTest(ExampleEndpoint.class)
class MyWebServiceServerTests {
@Autowired
private MockWebServiceClient client;
@Test
void mockServerCall() {
this.client
.sendRequest(RequestCreators.withPayload(new
StringSource("<ExampleRequest/>")))
.andExpect(ResponseMatchers.payload(new
StringSource("<ExampleResponse>42</ExampleResponse>")));
}
202
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import
org.springframework.boot.test.autoconfigure.webservices.server.WebServiceServerTest
import org.springframework.ws.test.server.MockWebServiceClient
import org.springframework.ws.test.server.RequestCreators
import org.springframework.ws.test.server.ResponseMatchers
import org.springframework.xml.transform.StringSource
@WebServiceServerTest(ExampleEndpoint::class)
class MyWebServiceServerTests(@Autowired val client: MockWebServiceClient) {
@Test
fun mockServerCall() {
client
.sendRequest(RequestCreators.withPayload(StringSource("<ExampleRequest/>")))
.andExpect(ResponseMatchers.payload(StringSource("<ExampleResponse>42</ExampleResponse
>")))
}
Each slice provides one or more @AutoConfigure… annotations that namely defines the auto-
configurations that should be included as part of a slice. Additional auto-configurations can be
added on a test-by-test basis by creating a custom @AutoConfigure… annotation or by adding
@ImportAutoConfiguration to the test as shown in the following example:
Java
import org.springframework.boot.autoconfigure.ImportAutoConfiguration;
import
org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration;
import org.springframework.boot.test.autoconfigure.jdbc.JdbcTest;
@JdbcTest
@ImportAutoConfiguration(IntegrationAutoConfiguration.class)
class MyJdbcTests {
203
Kotlin
import org.springframework.boot.autoconfigure.ImportAutoConfiguration
import org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration
import org.springframework.boot.test.autoconfigure.jdbc.JdbcTest
@JdbcTest
@ImportAutoConfiguration(IntegrationAutoConfiguration::class)
class MyJdbcTests
Make sure to not use the regular @Import annotation to import auto-configurations
NOTE
as they are handled in a specific way by Spring Boot.
Alternatively, additional auto-configurations can be added for any use of a slice annotation by
registering them in a file stored in META-INF/spring as shown in the following example:
META-INF/spring/org.springframework.boot.test.autoconfigure.jdbc.JdbcTest.imports
com.example.IntegrationAutoConfiguration
If you structure your code in a sensible way, your @SpringBootApplication class is used by default as
the configuration of your tests.
It then becomes important not to litter the application’s main class with configuration settings that
are specific to a particular area of its functionality.
Assume that you are using Spring Batch and you rely on the auto-configuration for it. You could
define your @SpringBootApplication as follows:
204
Java
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
@EnableBatchProcessing
public class MyApplication {
// ...
Kotlin
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing
import org.springframework.boot.autoconfigure.SpringBootApplication
@SpringBootApplication
@EnableBatchProcessing
class MyApplication {
// ...
Because this class is the source configuration for the test, any slice test actually tries to start Spring
Batch, which is definitely not what you want to do. A recommended approach is to move that area-
specific configuration to a separate @Configuration class at the same level as your application, as
shown in the following example:
Java
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
@EnableBatchProcessing
public class MyBatchConfiguration {
// ...
205
Kotlin
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
@EnableBatchProcessing
class MyBatchConfiguration {
// ...
Depending on the complexity of your application, you may either have a single
@Configuration class for your customizations or one class per domain area. The
NOTE latter approach lets you enable it in one of your tests, if necessary, with the @Import
annotation. See this how-to section for more details on when you might want to
enable specific @Configuration classes for slice tests.
Test slices exclude @Configuration classes from scanning. For example, for a @WebMvcTest, the
following configuration will not include the given WebMvcConfigurer bean in the application context
loaded by the test slice:
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration(proxyBeanMethods = false)
public class MyWebConfiguration {
@Bean
public WebMvcConfigurer testConfigurer() {
return new WebMvcConfigurer() {
// ...
};
}
206
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer
@Configuration(proxyBeanMethods = false)
class MyWebConfiguration {
@Bean
fun testConfigurer(): WebMvcConfigurer {
return object : WebMvcConfigurer {
// ...
}
}
The configuration below will, however, cause the custom WebMvcConfigurer to be loaded by the test
slice.
Java
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Component
public class MyWebMvcConfigurer implements WebMvcConfigurer {
// ...
Kotlin
import org.springframework.stereotype.Component
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer
@Component
class MyWebMvcConfigurer : WebMvcConfigurer {
// ...
Another source of confusion is classpath scanning. Assume that, while you structured your code in
a sensible way, you need to scan an additional package. Your application may resemble the
following code:
207
Java
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ComponentScan;
@SpringBootApplication
@ComponentScan({ "com.example.app", "com.example.another" })
public class MyApplication {
// ...
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.context.annotation.ComponentScan
@SpringBootApplication
@ComponentScan("com.example.app", "com.example.another")
class MyApplication {
// ...
Doing so effectively overrides the default component scan directive with the side effect of scanning
those two packages regardless of the slice that you chose. For instance, a @DataJpaTest seems to
suddenly scan components and user configurations of your application. Again, moving the custom
directive to a separate class is a good way to fix this issue.
If this is not an option for you, you can create a @SpringBootConfiguration somewhere
TIP in the hierarchy of your test so that it is used instead. Alternatively, you can specify a
source for your test, which disables the behavior of finding a default one.
Spock 2.2 or later can be used to test a Spring Boot application. To do so, add a dependency on a
-groovy-4.0 version of Spock’s spock-spring module to your application’s build. spock-spring
integrates Spring’s test framework into Spock. See the documentation for Spock’s Spring module for
further details.
7.8.4. Testcontainers
The Testcontainers library provides a way to manage services running inside Docker containers. It
integrates with JUnit, allowing you to write a test class that can start up a container before any of
the tests run. Testcontainers is especially useful for writing integration tests that talk to a real
backend service such as MySQL, MongoDB, Cassandra and others.
208
Testcontainers can be used in a Spring Boot test as follows:
Java
import org.junit.jupiter.api.Test;
import org.testcontainers.containers.Neo4jContainer;
import org.testcontainers.junit.jupiter.Container;
import org.testcontainers.junit.jupiter.Testcontainers;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest
@Testcontainers
class MyIntegrationTests {
@Container
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.boot.test.context.SpringBootTest
import org.testcontainers.containers.Neo4jContainer
import org.testcontainers.junit.jupiter.Container
import org.testcontainers.junit.jupiter.Testcontainers
@SpringBootTest
@Testcontainers
internal class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
var neo4j: Neo4jContainer<*> = Neo4jContainer<Nothing>("neo4j:5")
}
This will start up a docker container running Neo4j (if Docker is running locally) before any of the
209
tests are run. In most cases, you will need to configure the application to connect to the service
running in the container.
Service Connections
A service connection is a connection to any remote service. Spring Boot’s auto-configuration can
consume the details of a service connection and use them to establish a connection to a remote
service. When doing so, the connection details take precedence over any connection-related
configuration properties.
When using Testcontainers, connection details can be automatically created for a service running
in a container by annotating the container field in the test class.
Java
import org.junit.jupiter.api.Test;
import org.testcontainers.containers.Neo4jContainer;
import org.testcontainers.junit.jupiter.Container;
import org.testcontainers.junit.jupiter.Testcontainers;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.testcontainers.service.connection.ServiceConnection;
@SpringBootTest
@Testcontainers
class MyIntegrationTests {
@Container
@ServiceConnection
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
210
Kotlin
import org.junit.jupiter.api.Test
import org.testcontainers.containers.Neo4jContainer
import org.testcontainers.junit.jupiter.Container
import org.testcontainers.junit.jupiter.Testcontainers
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.testcontainers.service.connection.ServiceConnection;
@SpringBootTest
@Testcontainers
internal class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
@ServiceConnection
var neo4j: Neo4jContainer<*> = Neo4jContainer<Nothing>("neo4j:5")
The following service connection factories are provided in the spring-boot-testcontainers jar:
211
Connection Details Matched on
FlywayConnectionDetails Containers of type JdbcDatabaseContainer
JdbcConnectionDetails Containers of type JdbcDatabaseContainer
KafkaConnectionDetails Containers of type KafkaContainer or
RedpandaContainer
LiquibaseConnectionDetails Containers of type JdbcDatabaseContainer
MongoConnectionDetails Containers of type MongoDBContainer
Neo4jConnectionDetails Containers of type Neo4jContainer
R2dbcConnectionDetails Containers of type MariaDBContainer,
MSSQLServerContainer, MySQLContainer,
OracleContainer, or PostgreSQLContainer
RabbitConnectionDetails Containers of type RabbitMQContainer
RedisConnectionDetails Containers named "redis"
ZipkinConnectionDetails Containers named "openzipkin/zipkin"
By default all applicable connection details beans will be created for a given Container.
For example, a PostgreSQLContainer will create both JdbcConnectionDetails and
R2dbcConnectionDetails.
TIP
If you want to create only a subset of the applicable types, you can use the type
attribute of @ServiceConnection.
Dynamic Properties
A slightly more verbose but also more flexible alternative to service connections is
@DynamicPropertySource. A static @DynamicPropertySource method allows adding dynamic property
values to the Spring Environment.
212
Java
import org.junit.jupiter.api.Test;
import org.testcontainers.containers.Neo4jContainer;
import org.testcontainers.junit.jupiter.Container;
import org.testcontainers.junit.jupiter.Testcontainers;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.DynamicPropertyRegistry;
import org.springframework.test.context.DynamicPropertySource;
@SpringBootTest
@Testcontainers
class MyIntegrationTests {
@Container
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
@DynamicPropertySource
static void neo4jProperties(DynamicPropertyRegistry registry) {
registry.add("spring.neo4j.uri", neo4j::getBoltUrl);
}
213
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.test.context.DynamicPropertyRegistry
import org.springframework.test.context.DynamicPropertySource
import org.testcontainers.containers.Neo4jContainer
import org.testcontainers.junit.jupiter.Container
import org.testcontainers.junit.jupiter.Testcontainers
@SpringBootTest
@Testcontainers
internal class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
var neo4j: Neo4jContainer<*> = Neo4jContainer<Nothing>("neo4j:5")
@DynamicPropertySource
fun neo4jProperties(registry: DynamicPropertyRegistry) {
registry.add("spring.neo4j.uri") { neo4j.boltUrl }
}
The above configuration allows Neo4j-related beans in the application to communicate with Neo4j
running inside the Testcontainers-managed Docker container.
As well as using Testcontainers for integration testing, it’s also possible to use them at development
time. This approach allows developers to quickly start containers for the services that the
application depends on, removing the need to manually provision things like database servers.
Using Testcontainers in this way provides functionality similar to Docker Compose, except that your
container configuration is in Java rather than YAML.
To use Testcontainers at development time you need to launch your application using your “test”
classpath rather than “main”. This will allow you to access all declared test dependencies and give
you a natural place to write your test configuration.
To create a test launchable version of your application you should create an “Application” class in
the src/test directory. For example, if your main application is in
214
src/main/java/com/example/MyApplication.java, you should create
src/test/java/com/example/TestMyApplication.java
The TestMyApplication class can use the SpringApplication.from(…) method to launch the real
application:
Java
import org.springframework.boot.SpringApplication;
Kotlin
import org.springframework.boot.SpringApplication
import
org.springframework.boot.docs.features.testing.testcontainers.atdevelopmenttime.launch
.main as myApplicationMain
object Main {
@JvmStatic
fun main(args: Array<String>) {
SpringApplication.from(::myApplicationMain).run(*args)
}
}
You’ll also need to define the Container instances that you want to start along with your application.
To do this, you need to make sure that the spring-boot-testcontainers module has been added as a
test dependency. Once that has been done, you can create a @TestConfiguration class that declares
@Bean methods for the containers you want to start.
You can also annotate your @Bean methods with @ServiceConnection in order to create
ConnectionDetails beans. See the service connections section above for details of the supported
technologies.
215
Java
import org.testcontainers.containers.Neo4jContainer;
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.boot.testcontainers.service.connection.ServiceConnection;
import org.springframework.context.annotation.Bean;
@TestConfiguration(proxyBeanMethods = false)
public class MyContainersConfiguration {
@Bean
@ServiceConnection
public Neo4jContainer<?> neo4jContainer() {
return new Neo4jContainer<>("neo4j:5");
}
Kotlin
import org.testcontainers.containers.Neo4jContainer
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.boot.testcontainers.service.connection.ServiceConnection
import org.springframework.context.annotation.Bean;
@TestConfiguration(proxyBeanMethods = false)
class MyContainersConfiguration {
@Bean
@ServiceConnection
fun neo4jContainer(): Neo4jContainer<*> {
return Neo4jContainer("neo4j:5")
}
Once you have defined your test configuration, you can use the with(…) method to attach it to your
test launcher:
216
Java
import org.springframework.boot.SpringApplication;
SpringApplication.from(MyApplication::main).with(MyContainersConfiguration.class).run(
args);
}
Kotlin
import org.springframework.boot.SpringApplication
import
org.springframework.boot.docs.features.testing.testcontainers.atdevelopmenttime.test.m
ain as myApplicationMain
object Main {
@JvmStatic
fun main(args: Array<String>) {
SpringApplication.from(::myApplicationMain).with(MyContainersConfiguration::class.java
).run(*args)
}
}
You can now launch TestMyApplication as you would any regular Java main method application to
start your application and the containers that it needs to run.
If you want to contribute dynamic properties at development time from your Container @Bean
methods, you can do so by injecting a DynamicPropertyRegistry. This works in a similar way to the
@DynamicPropertySource annotation that you can use in your tests. It allows you to add properties
that will become available once your container has started.
217
Java
import org.testcontainers.containers.MongoDBContainer;
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.test.context.DynamicPropertyRegistry;
@TestConfiguration(proxyBeanMethods = false)
public class MyContainersConfiguration {
@Bean
public MongoDBContainer monogDbContainer(DynamicPropertyRegistry properties) {
MongoDBContainer container = new MongoDBContainer("mongo:5.0");
properties.add("spring.data.mongodb.host", container::getHost);
properties.add("spring.data.mongodb.port", container::getFirstMappedPort);
return container;
}
Kotlin
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.context.annotation.Bean;
import org.springframework.test.context.DynamicPropertyRegistry
import org.testcontainers.containers.MongoDBContainer
@TestConfiguration(proxyBeanMethods = false)
class MyContainersConfiguration {
@Bean
fun monogDbContainer(properties: DynamicPropertyRegistry): MongoDBContainer {
var container = MongoDBContainer("mongo:5.0")
properties.add("spring.data.mongodb.host", container::getHost);
properties.add("spring.data.mongodb.port", container::getFirstMappedPort);
return container
}
A common pattern when using Testcontainers is to declare Container instances as static fields. Often
these fields are defined directly on the test class. They can also be declared on a parent class or on
218
an interface that the test implements.
For example, the following MyContainers interface declares mongo and neo4j containers:
import org.testcontainers.containers.MongoDBContainer;
import org.testcontainers.containers.Neo4jContainer;
import org.testcontainers.junit.jupiter.Container;
@Container
MongoDBContainer monogContainer = new MongoDBContainer("mongo:5.0");
@Container
Neo4jContainer<?> neo4jContainer = new Neo4jContainer<>("neo4j:5");
If you already have containers defined in this way, or you just prefer this style, you can import
these declaration classes rather than defining you containers as @Bean methods. To do so, add the
@ImportTestcontainers annotation to your test configuration class:
Java
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.boot.testcontainers.context.ImportTestcontainers;
@TestConfiguration(proxyBeanMethods = false)
@ImportTestcontainers(MyContainers.class)
public class MyContainersConfiguration {
Kotlin
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.boot.testcontainers.context.ImportTestcontainers
@TestConfiguration(proxyBeanMethods = false)
@ImportTestcontainers(MyContainers::class)
class MyContainersConfiguration {
You can use the @ServiceConnection annotation on Container fields to establish service
TIP connections. You can also add @DynamicPropertySource annotated methods to your
declaration class.
219
7.8.5. Test Utilities
A few test utility classes that are generally useful when testing your application are packaged as
part of spring-boot.
ConfigDataApplicationContextInitializer
Java
import org.springframework.boot.test.context.ConfigDataApplicationContextInitializer;
import org.springframework.test.context.ContextConfiguration;
// ...
Kotlin
import org.springframework.boot.test.context.ConfigDataApplicationContextInitializer
import org.springframework.test.context.ContextConfiguration
// ...
TestPropertyValues
220
Java
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.util.TestPropertyValues;
import org.springframework.mock.env.MockEnvironment;
class MyEnvironmentTests {
@Test
void testPropertySources() {
MockEnvironment environment = new MockEnvironment();
TestPropertyValues.of("org=Spring", "name=Boot").applyTo(environment);
assertThat(environment.getProperty("name")).isEqualTo("Boot");
}
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.boot.test.util.TestPropertyValues
import org.springframework.mock.env.MockEnvironment
class MyEnvironmentTests {
@Test
fun testPropertySources() {
val environment = MockEnvironment()
TestPropertyValues.of("org=Spring", "name=Boot").applyTo(environment)
assertThat(environment.getProperty("name")).isEqualTo("Boot")
}
OutputCapture
OutputCapture is a JUnit Extension that you can use to capture System.out and System.err output. To
use it, add @ExtendWith(OutputCaptureExtension.class) and inject CapturedOutput as an argument to
your test class constructor or test method as follows:
221
Java
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.springframework.boot.test.system.CapturedOutput;
import org.springframework.boot.test.system.OutputCaptureExtension;
@ExtendWith(OutputCaptureExtension.class)
class MyOutputCaptureTests {
@Test
void testName(CapturedOutput output) {
System.out.println("Hello World!");
assertThat(output).contains("World");
}
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.extension.ExtendWith
import org.springframework.boot.test.system.CapturedOutput
import org.springframework.boot.test.system.OutputCaptureExtension
@ExtendWith(OutputCaptureExtension::class)
class MyOutputCaptureTests {
@Test
fun testName(output: CapturedOutput?) {
println("Hello World!")
assertThat(output).contains("World")
}
TestRestTemplate
222
Spring Framework 5.0 provides a new WebTestClient that works for WebFlux
TIP integration tests and both WebFlux and MVC end-to-end testing. It provides a fluent
API for assertions, unlike TestRestTemplate.
It is recommended, but not mandatory, to use the Apache HTTP Client (version 5.1 or better). If you
have that on your classpath, the TestRestTemplate responds by configuring the client appropriately.
If you do use Apache’s HTTP client, some additional test-friendly features are enabled:
• Redirects are not followed (so you can assert the response location).
TestRestTemplate can be instantiated directly in your integration tests, as shown in the following
example:
Java
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.web.client.TestRestTemplate;
import org.springframework.http.ResponseEntity;
class MyTests {
@Test
void testRequest() {
ResponseEntity<String> headers =
this.template.getForEntity("https://myhost.example.com/example", String.class);
assertThat(headers.getHeaders().getLocation()).hasHost("other.example.com");
}
223
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.boot.test.web.client.TestRestTemplate
class MyTests {
@Test
fun testRequest() {
val headers = template.getForEntity("https://myhost.example.com/example",
String::class.java)
assertThat(headers.headers.location).hasHost("other.example.com")
}
224
Java
import java.time.Duration;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.boot.test.context.TestConfiguration;
import org.springframework.boot.test.web.client.TestRestTemplate;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.http.HttpHeaders;
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MySpringBootTests {
@Autowired
private TestRestTemplate template;
@Test
void testRequest() {
HttpHeaders headers = this.template.getForEntity("/example",
String.class).getHeaders();
assertThat(headers.getLocation()).hasHost("other.example.com");
}
@TestConfiguration(proxyBeanMethods = false)
static class RestTemplateBuilderConfiguration {
@Bean
RestTemplateBuilder restTemplateBuilder() {
return new RestTemplateBuilder().setConnectTimeout(Duration.ofSeconds(1))
.setReadTimeout(Duration.ofSeconds(1));
}
225
Kotlin
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment
import org.springframework.boot.test.context.TestConfiguration
import org.springframework.boot.test.web.client.TestRestTemplate
import org.springframework.boot.web.client.RestTemplateBuilder
import org.springframework.context.annotation.Bean
import java.time.Duration
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MySpringBootTests(@Autowired val template: TestRestTemplate) {
@Test
fun testRequest() {
val headers = template.getForEntity("/example", String::class.java).headers
assertThat(headers.location).hasHost("other.example.com")
}
@TestConfiguration(proxyBeanMethods = false)
internal class RestTemplateBuilderConfiguration {
@Bean
fun restTemplateBuilder(): RestTemplateBuilder {
return RestTemplateBuilder().setConnectTimeout(Duration.ofSeconds(1))
.setReadTimeout(Duration.ofSeconds(1))
}
A typical workflow with Docker Compose is to run docker compose up. Work on your application
with it connecting to started services. Then run docker compose down when you are finished.
To help with this workflow, the spring-boot-docker-compose module can be used. When this modules
is included as a dependency Spring Boot will do the following:
• Search for a compose.yml and other common compose filenames in your application directory
226
• Create service connection beans for each supported container
A service connection is a connection to any remote service. Spring Boot’s auto-configuration can
consume the details of a service connection and use them to establish a connection to a remote
service. When doing so, the connection details take precedence over any connection-related
configuration properties.
When using Spring Boot’s Docker Compose support, service connections are established to the port
mapped by the container.
Docker compose is usually used in such a way that the ports inside the container are
mapped to ephemeral ports on your computer. For example, A Postgres server my
NOTE run inside the container using port 5432 but be mapped to a totally different port
locally. The service connection will always discover and use the locally mapped
port.
Service connections are established by using the image name of the container. The following
service connections are currently supported:
Sometimes you may need to use your own version of an image to provide a service. You can use any
custom image as long as it behaves in the same way as the standard image. Specifically, any
environment variables that the standard image supports must also be used in your custom image.
If your image uses a different name, you can use a label in your compose.yml file so that Spring Boot
227
can provide a service connection. Use a label named org.springframework.boot.service-connection
to provide the service name.
For example:
services:
redis:
image: 'mycompany/mycustomredis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.service-connection: redis
If you have a container image defined in your compose.yml that you don’t want connected to your
application you can use a label to ignore it. Any container with labeled with
org.springframework.boot.ignore will be ignored by Spring Boot.
For example:
services:
redis:
image: 'redis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.ignore: true
If your compose file is not in the same directory as your application, or if it’s named differently, you
can use spring.docker.compose.file in your application.properties or application.yaml to point to a
different file. Properties can be defined as an exact path or a path that’s relative to your
application.
For example:
Properties
spring.docker.compose.file=../my-compose.yml
228
Yaml
spring:
docker:
compose:
file: "../my-compose.yml"
Containers started by Docker Compose may take some time to become fully ready. The
recommended way of checking for readiness is to add a healthcheck section under the service
definition in your compose.yml file.
Since it’s not uncommon for healthcheck configuration to be omitted from compose.yml files, Spring
Boot also checks directly for service readiness. By default, a container is considered ready when a
TCP/IP connection can be established to its mapped port.
For example:
services:
redis:
image: 'redis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.readiness-check.tcp.disable: true
You can also change timeout values in your application.properties or application.yaml file:
Properties
spring.docker.compose.readiness.tcp.connect-timeout=10s
spring.docker.compose.readiness.tcp.read-timeout=5s
Yaml
spring:
docker:
compose:
readiness:
tcp:
connect-timeout: 10s
read-timeout: 5s
229
You can also provide your own ServiceReadinessCheck implementations and register
TIP
them in the spring.factories file.
By default Spring Boot calls docker compose up when your application starts and docker compose
stop when it’s shut down. If you prefer to have different lifecycle management you can use the
spring.docker.compose.lifecycle-management property.
• start-only - Start Docker Compose when the application starts and leave it running
• start-and-stop - Start Docker Compose whe the application starts and stop it when the JVM exits
Properties
spring.docker.compose.lifecycle-management=start-and-stop
spring.docker.compose.start.command=start
spring.docker.compose.stop.command=down
spring.docker.compose.stop.timeout=1m
Yaml
spring:
docker:
compose:
lifecycle-management: start-and-stop
start:
command: start
stop:
command: down
timeout: 1m
Docker Compose profiles are similar to Spring profiles in that they let you adjust your Docker
Compose configuration for specific environments. If you want to activate a specific Docker
Compose profile you can use the spring.docker.compose.profiles.active property in your
application.properties or application.yaml file:
230
Properties
spring.docker.compose.profiles.active=myprofile
Yaml
spring:
docker:
compose:
profiles:
active: "myprofile"
Classes that implement auto-configuration are annotated with @AutoConfiguration. This annotation
itself is meta-annotated with @Configuration, making auto-configurations standard @Configuration
classes. Additional @Conditional annotations are used to constrain when the auto-configuration
should apply. Usually, auto-configuration classes use @ConditionalOnClass and
@ConditionalOnMissingBean annotations. This ensures that auto-configuration applies only when
relevant classes are found and when you have not declared your own @Configuration.
You can browse the source code of spring-boot-autoconfigure to see the @AutoConfiguration classes
that Spring provides (see the META-
INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports file).
com.mycorp.libx.autoconfigure.LibXAutoConfiguration
com.mycorp.libx.autoconfigure.LibXWebAutoConfiguration
231
TIP You can add comments to the imports file using the # character.
Auto-configurations must be loaded only by being named in the imports file. Make
sure that they are defined in a specific package space and that they are never the
NOTE target of component scanning. Furthermore, auto-configuration classes should not
enable component scanning to find additional components. Specific @Import
annotations should be used instead.
If your configuration needs to be applied in a specific order, you can use the before, beforeName,
after and afterName attributes on the @AutoConfiguration annotation or the dedicated
@AutoConfigureBefore and @AutoConfigureAfter annotations. For example, if you provide web-
specific configuration, your class may need to be applied after WebMvcAutoConfiguration.
If you want to order certain auto-configurations that should not have any direct knowledge of each
other, you can also use @AutoConfigureOrder. That annotation has the same semantic as the regular
@Order annotation but provides a dedicated order for auto-configuration classes.
As with standard @Configuration classes, the order in which auto-configuration classes are applied
only affects the order in which their beans are defined. The order in which those beans are
subsequently created is unaffected and is determined by each bean’s dependencies and any
@DependsOn relationships.
You almost always want to include one or more @Conditional annotations on your auto-
configuration class. The @ConditionalOnMissingBean annotation is one common example that is used
to allow developers to override auto-configuration if they are not happy with your defaults.
Spring Boot includes a number of @Conditional annotations that you can reuse in your own code by
annotating @Configuration classes or individual @Bean methods. These annotations include:
• Class Conditions
• Bean Conditions
• Property Conditions
• Resource Conditions
Class Conditions
This mechanism does not apply the same way to @Bean methods where typically the return type is
232
the target of the condition: before the condition on the method applies, the JVM will have loaded
the class and potentially processed method references which will fail if the class is not present.
To handle this scenario, a separate @Configuration class can be used to isolate the condition, as
shown in the following example:
Java
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@AutoConfiguration
// Some conditions ...
public class MyAutoConfiguration {
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(SomeService.class)
public static class SomeServiceConfiguration {
@Bean
@ConditionalOnMissingBean
public SomeService someService() {
return new SomeService();
}
233
Kotlin
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
// Some conditions ...
class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
fun someService(): SomeService {
return SomeService()
}
Bean Conditions
When placed on a @Bean method, the target type defaults to the return type of the method, as shown
in the following example:
234
Java
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
@AutoConfiguration
public class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
public SomeService someService() {
return new SomeService();
}
Kotlin
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
fun someService(): SomeService {
return SomeService()
}
In the preceding example, the someService bean is going to be created if no bean of type SomeService
is already contained in the ApplicationContext.
You need to be very careful about the order in which bean definitions are added, as
these conditions are evaluated based on what has been processed so far. For this
TIP reason, we recommend using only @ConditionalOnBean and @ConditionalOnMissingBean
annotations on auto-configuration classes (since these are guaranteed to load after
any user-defined bean definitions have been added).
235
When declaring a @Bean method, provide as much type information as possible in the
method’s return type. For example, if your bean’s concrete class implements an
interface the bean method’s return type should be the concrete class and not the
TIP
interface. Providing as much type information as possible in @Bean methods is
particularly important when using bean conditions as their evaluation can only rely
upon to type information that is available in the method signature.
Property Conditions
Resource Conditions
The @ConditionalOnResource annotation lets configuration be included only when a specific resource
is present. Resources can be specified by using the usual Spring conventions, as shown in the
following example: file:/home/user/test.dat.
Referencing a bean in the expression will cause that bean to be initialized very
early in context refresh processing. As a result, the bean won’t be eligible for post-
NOTE
processing (such as configuration properties binding) and its state may be
incomplete.
An auto-configuration can be affected by many factors: user configuration (@Bean definition and
Environment customization), condition evaluation (presence of a particular library), and others.
Concretely, each test should create a well defined ApplicationContext that represents a combination
of those customizations. ApplicationContextRunner provides a great way to achieve that.
236
ApplicationContextRunner is usually defined as a field of the test class to gather the base, common
configuration. The following example makes sure that MyServiceAutoConfiguration is always
invoked:
Java
Kotlin
Each test can use the runner to represent a particular use case. For instance, the sample below
invokes a user configuration (UserConfiguration) and checks that the auto-configuration backs off
properly. Invoking run provides a callback context that can be used with AssertJ.
Java
@Test
void defaultServiceBacksOff() {
this.contextRunner.withUserConfiguration(UserConfiguration.class).run((context) ->
{
assertThat(context).hasSingleBean(MyService.class);
assertThat(context).getBean("myCustomService").isSameAs(context.getBean(MyService.clas
s));
});
}
@Configuration(proxyBeanMethods = false)
static class UserConfiguration {
@Bean
MyService myCustomService() {
return new MyService("mine");
}
237
Kotlin
@Test
fun defaultServiceBacksOff() {
contextRunner.withUserConfiguration(UserConfiguration::class.java)
.run { context: AssertableApplicationContext ->
assertThat(context).hasSingleBean(MyService::class.java)
assertThat(context).getBean("myCustomService")
.isSameAs(context.getBean(MyService::class.java))
}
}
@Configuration(proxyBeanMethods = false)
internal class UserConfiguration {
@Bean
fun myCustomService(): MyService {
return MyService("mine")
}
It is also possible to easily customize the Environment, as shown in the following example:
Java
@Test
void serviceNameCanBeConfigured() {
this.contextRunner.withPropertyValues("user.name=test123").run((context) -> {
assertThat(context).hasSingleBean(MyService.class);
assertThat(context.getBean(MyService.class).getName()).isEqualTo("test123");
});
}
Kotlin
@Test
fun serviceNameCanBeConfigured() {
contextRunner.withPropertyValues("user.name=test123").run { context:
AssertableApplicationContext ->
assertThat(context).hasSingleBean(MyService::class.java)
assertThat(context.getBean(MyService::class.java).name).isEqualTo("test123")
}
}
The runner can also be used to display the ConditionEvaluationReport. The report can be printed at
INFO or DEBUG level. The following example shows how to use the
ConditionEvaluationReportLoggingListener to print the report in auto-configuration tests.
238
Java
import org.junit.jupiter.api.Test;
import
org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListene
r;
import org.springframework.boot.logging.LogLevel;
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
class MyConditionEvaluationReportingTests {
@Test
void autoConfigTest() {
new ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run((context) -> {
// Test something...
});
}
Kotlin
import org.junit.jupiter.api.Test
import
org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListene
r
import org.springframework.boot.logging.LogLevel
import org.springframework.boot.test.context.assertj.AssertableApplicationContext
import org.springframework.boot.test.context.runner.ApplicationContextRunner
class MyConditionEvaluationReportingTests {
@Test
fun autoConfigTest() {
ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run { context: AssertableApplicationContext? -> }
}
If you need to test an auto-configuration that only operates in a servlet or reactive web application
context, use the WebApplicationContextRunner or ReactiveWebApplicationContextRunner respectively.
239
Overriding the Classpath
It is also possible to test what happens when a particular class and/or package is not present at
runtime. Spring Boot ships with a FilteredClassLoader that can easily be used by the runner. In the
following example, we assert that if MyService is not present, the auto-configuration is properly
disabled:
Java
@Test
void serviceIsIgnoredIfLibraryIsNotPresent() {
this.contextRunner.withClassLoader(new FilteredClassLoader(MyService.class))
.run((context) -> assertThat(context).doesNotHaveBean("myService"));
}
Kotlin
@Test
fun serviceIsIgnoredIfLibraryIsNotPresent() {
contextRunner.withClassLoader(FilteredClassLoader(MyService::class.java))
.run { context: AssertableApplicationContext? ->
assertThat(context).doesNotHaveBean("myService")
}
}
A typical Spring Boot starter contains code to auto-configure and customize the infrastructure of a
given technology, let’s call that "acme". To make it easily extensible, a number of configuration keys
in a dedicated namespace can be exposed to the environment. Finally, a single "starter" dependency
is provided to help users get started as easily as possible.
• The autoconfigure module that contains the auto-configuration code for "acme".
• The starter module that provides a dependency to the autoconfigure module as well as "acme"
and any additional dependencies that are typically useful. In a nutshell, adding the starter
should provide everything needed to start using that library.
This separation in two modules is in no way necessary. If "acme" has several flavors, options or
optional features, then it is better to separate the auto-configuration as you can clearly express the
fact some features are optional. Besides, you have the ability to craft a starter that provides an
opinion about those optional dependencies. At the same time, others can rely only on the
autoconfigure module and craft their own starter with different opinions.
If the auto-configuration is relatively straightforward and does not have optional features, merging
the two modules in the starter is definitely an option.
240
Naming
You should make sure to provide a proper namespace for your starter. Do not start your module
names with spring-boot, even if you use a different Maven groupId. We may offer official support
for the thing you auto-configure in the future.
As a rule of thumb, you should name a combined module after the starter. For example, assume
that you are creating a starter for "acme" and that you name the auto-configure module acme-
spring-boot and the starter acme-spring-boot-starter. If you only have one module that combines
the two, name it acme-spring-boot-starter.
Configuration keys
If your starter provides configuration keys, use a unique namespace for them. In particular, do not
include your keys in the namespaces that Spring Boot uses (such as server, management, spring, and
so on). If you use the same namespace, we may modify these namespaces in the future in ways that
break your modules. As a rule of thumb, prefix all your keys with a namespace that you own (for
example acme).
Make sure that configuration keys are documented by adding field javadoc for each property, as
shown in the following example:
241
Java
import java.time.Duration;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("acme")
public class AcmeProperties {
/**
* Whether to check the location of acme resources.
*/
private boolean checkLocation = true;
/**
* Timeout for establishing a connection to the acme server.
*/
private Duration loginTimeout = Duration.ofSeconds(3);
242
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import java.time.Duration
@ConfigurationProperties("acme")
class AcmeProperties(
/**
* Whether to check the location of acme resources.
*/
var isCheckLocation: Boolean = true,
/**
* Timeout for establishing a connection to the acme server.
*/
var loginTimeout:Duration = Duration.ofSeconds(3))
You should only use plain text with @ConfigurationProperties field Javadoc, since
NOTE
they are not processed before being added to the JSON.
Here are some rules we follow internally to make sure descriptions are consistent:
• Use java.time.Duration rather than long and describe the default unit if it differs from
milliseconds, such as "If a duration suffix is not specified, seconds will be used".
• Do not provide the default value in the description unless it has to be determined at runtime.
Make sure to trigger meta-data generation so that IDE assistance is available for your keys as well.
You may want to review the generated metadata (META-INF/spring-configuration-metadata.json) to
make sure your keys are properly documented. Using your own starter in a compatible IDE is also a
good idea to validate that quality of the metadata.
The autoconfigure module contains everything that is necessary to get started with the library. It
may also contain configuration key definitions (such as @ConfigurationProperties) and any callback
interface that can be used to further customize how the components are initialized.
You should mark the dependencies to the library as optional so that you can include
TIP the autoconfigure module in your projects more easily. If you do it that way, the library
is not provided and, by default, Spring Boot backs off.
243
to eagerly filter auto-configurations that do not match, which will improve startup time.
When building with Maven, it is recommended to add the following dependency in a module that
contains auto-configurations:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
<optional>true</optional>
</dependency>
If you have defined auto-configurations directly in your application, make sure to configure the
spring-boot-maven-plugin to prevent the repackage goal from adding the dependency into the fat jar:
<project>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<excludes>
<exclude>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-
processor</artifactId>
</exclude>
</excludes>
</configuration>
</plugin>
</plugins>
</build>
</project>
dependencies {
annotationProcessor "org.springframework.boot:spring-boot-autoconfigure-processor"
}
Starter Module
The starter is really an empty jar. Its only purpose is to provide the necessary dependencies to work
with the library. You can think of it as an opinionated view of what is required to get started.
Do not make assumptions about the project in which your starter is added. If the library you are
244
auto-configuring typically requires other starters, mention them as well. Providing a proper set of
default dependencies may be hard if the number of optional dependencies is high, as you should
avoid including dependencies that are unnecessary for a typical usage of the library. In other
words, you should not include optional dependencies.
Either way, your starter must reference the core Spring Boot starter (spring-boot-
starter) directly or indirectly (there is no need to add it if your starter relies on
NOTE
another starter). If a project is created with only your custom starter, Spring Boot’s
core features will be honoured by the presence of the core starter.
Spring Boot provides Kotlin support by leveraging the support in other Spring projects such as
Spring Framework, Spring Data, and Reactor. See the Spring Framework Kotlin support
documentation for more information.
The easiest way to start with Spring Boot and Kotlin is to follow this comprehensive tutorial. You
can create new Kotlin projects by using start.spring.io. Feel free to join the #spring channel of Kotlin
Slack or ask a question with the spring and kotlin tags on Stack Overflow if you need support.
7.11.1. Requirements
Spring Boot requires at least Kotlin 1.7.x and manages a suitable Kotlin version through
dependency management. To use Kotlin, org.jetbrains.kotlin:kotlin-stdlib and
org.jetbrains.kotlin:kotlin-reflect must be present on the classpath. The kotlin-stdlib variants
kotlin-stdlib-jdk7 and kotlin-stdlib-jdk8 can also be used.
Since Kotlin classes are final by default, you are likely to want to configure kotlin-spring plugin in
order to automatically open Spring-annotated classes so that they can be proxied.
Jackson’s Kotlin module is required for serializing / deserializing JSON data in Kotlin. It is
automatically registered when found on the classpath. A warning message is logged if Jackson and
Kotlin are present but the Jackson Kotlin module is not.
These dependencies and plugins are provided by default if one bootstraps a Kotlin
TIP
project on start.spring.io.
7.11.2. Null-safety
One of Kotlin’s key features is null-safety. It deals with null values at compile time rather than
deferring the problem to runtime and encountering a NullPointerException. This helps to eliminate
a common source of bugs without paying the cost of wrappers like Optional. Kotlin also allows using
functional constructs with nullable values as described in this comprehensive guide to null-safety
in Kotlin.
245
Although Java does not allow one to express null-safety in its type system, Spring Framework,
Spring Data, and Reactor now provide null-safety of their API through tooling-friendly annotations.
By default, types from Java APIs used in Kotlin are recognized as platform types for which null-
checks are relaxed. Kotlin’s support for JSR 305 annotations combined with nullability annotations
provide null-safety for the related Spring API in Kotlin.
The JSR 305 checks can be configured by adding the -Xjsr305 compiler flag with the following
options: -Xjsr305={strict|warn|ignore}. The default behavior is the same as -Xjsr305=warn. The
strict value is required to have null-safety taken in account in Kotlin types inferred from Spring
API but should be used with the knowledge that Spring API nullability declaration could evolve
even between minor releases and more checks may be added in the future).
Generic type arguments, varargs and array elements nullability are not yet
WARNING supported. See SPR-15942 for up-to-date information. Also be aware that
Spring Boot’s own API is not yet annotated.
runApplication
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
@SpringBootApplication
class MyApplication
runApplication<MyApplication>(*args) {
setBannerMode(OFF)
}
Extensions
Kotlin extensions provide the ability to extend existing classes with additional functionality. The
Spring Boot Kotlin API makes use of these extensions to add new Kotlin specific conveniences to
existing APIs.
246
Spring Framework, are provided. Among other things, the extensions make it possible to take
advantage of Kotlin reified type parameters.
In order to avoid mixing different versions of Kotlin dependencies on the classpath, Spring Boot
imports the Kotlin BOM.
With Maven, the Kotlin version can be customized by setting the kotlin.version property and
plugin management is provided for kotlin-maven-plugin. With Gradle, the Spring Boot plugin
automatically aligns the kotlin.version with the version of the Kotlin plugin.
Spring Boot also manages the version of Coroutines dependencies by importing the Kotlin
Coroutines BOM. The version can be customized by setting the kotlin-coroutines.version property.
7.11.5. @ConfigurationProperties
@ConfigurationProperties when used in combination with constructor binding supports classes with
immutable val properties as shown in the following example:
@ConfigurationProperties("example.kotlin")
data class KotlinExampleProperties(
val name: String,
val description: String,
val myService: MyService) {
To generate your own metadata using the annotation processor, kapt should be
configured with the spring-boot-configuration-processor dependency. Note that some
TIP
features (such as detecting the default value or deprecated items) are not working due
to limitations in the model kapt provides.
7.11.6. Testing
While it is possible to use JUnit 4 to test Kotlin code, JUnit 5 is provided by default and is
recommended. JUnit 5 enables a test class to be instantiated once and reused for all of the class’s
tests. This makes it possible to use @BeforeAll and @AfterAll annotations on non-static methods,
which is a good fit for Kotlin.
247
To mock Kotlin classes, MockK is recommended. If you need the MockK equivalent of the Mockito
specific @MockBean and @SpyBean annotations, you can use SpringMockK which provides similar
@MockkBean and @SpykBean annotations.
7.11.7. Resources
Further reading
• Kotlin blog
• Awesome Kotlin
Examples
• spring-kotlin-fullstack: WebFlux Kotlin fullstack example with Kotlin2js for frontend instead of
JavaScript or TypeScript
• spring-kotlin-deepdive: a step by step migration for Boot 1.0 + Java to Boot 2.0 + Kotlin
7.12. SSL
Spring Boot provides the ability to configure SSL trust material that can be applied to several types
of connections in order to support secure communications. Configuration properties with the prefix
spring.ssl.bundle can be used to specify named sets of trust material and associated information.
Configuration properties with the prefix spring.ssl.bundle.jks can be used to configure bundles of
trust material created with the Java keytool utility and stored in Java KeyStore files in the JKS or
PKCS12 format. Each bundle has a user-provided name that can be used to reference the bundle.
When used to secure an embedded web server, a keystore is typically configured with a Java
248
KeyStore containing a certificate and private key as shown in this example:
Properties
spring.ssl.bundle.jks.mybundle.key.alias=application
spring.ssl.bundle.jks.mybundle.keystore.location=classpath:application.p12
spring.ssl.bundle.jks.mybundle.keystore.password=secret
spring.ssl.bundle.jks.mybundle.keystore.type=PKCS12
Yaml
spring:
ssl:
bundle:
jks:
mybundle:
key:
alias: "application"
keystore:
location: "classpath:application.p12"
password: "secret"
type: "PKCS12"
When used to secure a client-side connection, a truststore is typically configured with a Java
KeyStore containing the server certificate as shown in this example:
Properties
spring.ssl.bundle.jks.mybundle.truststore.location=classpath:server.p12
spring.ssl.bundle.jks.mybundle.truststore.password=secret
Yaml
spring:
ssl:
bundle:
jks:
mybundle:
truststore:
location: "classpath:server.p12"
password: "secret"
Configuration properties with the prefix spring.ssl.bundle.pem can be used to configure bundles of
trust material in the form of PEM-encoded text. Each bundle has a user-provided name that can be
249
used to reference the bundle.
When used to secure an embedded web server, a keystore is typically configured with a certificate
and private key as shown in this example:
Properties
spring.ssl.bundle.pem.mybundle.keystore.certificate=classpath:application.crt
spring.ssl.bundle.pem.mybundle.keystore.private-key=classpath:application.key
Yaml
spring:
ssl:
bundle:
pem:
mybundle:
keystore:
certificate: "classpath:application.crt"
private-key: "classpath:application.key"
When used to secure an embedded web server, a truststore is typically configured with the server
certificate as shown in this example:
Properties
spring.ssl.bundle.pem.mybundle.truststore.certificate=classpath:server.crt
Yaml
spring:
ssl:
bundle:
pem:
mybundle:
truststore:
certificate: "classpath:server.crt"
Once configured using properties, SSL bundles can be referred to by name in configuration
properties for various types of connections that are auto-configured by Spring Boot. See the
sections on embedded web servers, data technologies, and REST clients for further information.
250
7.12.4. Using SSL Bundles
Spring Boot auto-configures a bean of type SslBundles that provides access to each of the named
bundles configured using the spring.ssl.bundle properties.
An SslBundle can be retrieved from the auto-configured SslBundles bean and used to create objects
that are used to configure SSL connectivity in client libraries. The SslBundle provides a layered
approach of obtaining these SSL objects:
• getStores() provides access to the key store and trust store java.security.KeyStore instances as
well as any required key store password.
In addition, the SslBundle provides details about the key being used, the protocol to use and any
option that should be applied to the SSL engine.
The following example shows retrieving an SslBundle and using it to create an SSLContext:
Java
import javax.net.ssl.SSLContext;
import org.springframework.boot.ssl.SslBundle;
import org.springframework.boot.ssl.SslBundles;
import org.springframework.stereotype.Component;
@Component
public class MyComponent {
251
Kotlin
import org.springframework.boot.ssl.SslBundles
import org.springframework.stereotype.Component
@Component
class MyComponent(sslBundles: SslBundles) {
init {
val sslBundle = sslBundles.getBundle("mybundle")
val sslContext = sslBundle.createSslContext()
// do something with the created sslContext
}
If you are comfortable with Spring Boot’s core features, you can continue on and read about
production-ready features.
252
Chapter 8. Web
Spring Boot is well suited for web application development. You can create a self-contained HTTP
server by using embedded Tomcat, Jetty, Undertow, or Netty. Most web applications use the spring-
boot-starter-web module to get up and running quickly. You can also choose to build reactive web
applications by using the spring-boot-starter-webflux module.
If you have not yet developed a Spring Boot web application, you can follow the "Hello World!"
example in the Getting started section.
The Spring Web MVC framework (often referred to as “Spring MVC”) is a rich “model view
controller” web framework. Spring MVC lets you create special @Controller or @RestController
beans to handle incoming HTTP requests. Methods in your controller are mapped to HTTP by using
@RequestMapping annotations.
The following code shows a typical @RestController that serves JSON data:
253
Java
import java.util.List;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/users")
public class MyRestController {
@GetMapping("/{userId}")
public User getUser(@PathVariable Long userId) {
return this.userRepository.findById(userId).get();
}
@GetMapping("/{userId}/customers")
public List<Customer> getUserCustomers(@PathVariable Long userId) {
return
this.userRepository.findById(userId).map(this.customerRepository::findByUser).get();
}
@DeleteMapping("/{userId}")
public void deleteUser(@PathVariable Long userId) {
this.userRepository.deleteById(userId);
}
254
Kotlin
import org.springframework.web.bind.annotation.DeleteMapping
import org.springframework.web.bind.annotation.GetMapping
import org.springframework.web.bind.annotation.PathVariable
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController
@RestController
@RequestMapping("/users")
class MyRestController(private val userRepository: UserRepository, private val
customerRepository: CustomerRepository) {
@GetMapping("/{userId}")
fun getUser(@PathVariable userId: Long): User {
return userRepository.findById(userId).get()
}
@GetMapping("/{userId}/customers")
fun getUserCustomers(@PathVariable userId: Long): List<Customer> {
return
userRepository.findById(userId).map(customerRepository::findByUser).get()
}
@DeleteMapping("/{userId}")
fun deleteUser(@PathVariable userId: Long) {
userRepository.deleteById(userId)
}
“WebMvc.fn”, the functional variant, separates the routing configuration from the actual handling
of the requests, as shown in the following example:
255
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.servlet.function.RequestPredicate;
import org.springframework.web.servlet.function.RouterFunction;
import org.springframework.web.servlet.function.ServerResponse;
@Configuration(proxyBeanMethods = false)
public class MyRoutingConfiguration {
@Bean
public RouterFunction<ServerResponse> routerFunction(MyUserHandler userHandler) {
return route()
.GET("/{user}", ACCEPT_JSON, userHandler::getUser)
.GET("/{user}/customers", ACCEPT_JSON, userHandler::getUserCustomers)
.DELETE("/{user}", ACCEPT_JSON, userHandler::deleteUser)
.build();
}
256
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.http.MediaType
import org.springframework.web.servlet.function.RequestPredicates.accept
import org.springframework.web.servlet.function.RouterFunction
import org.springframework.web.servlet.function.RouterFunctions
import org.springframework.web.servlet.function.ServerResponse
@Configuration(proxyBeanMethods = false)
class MyRoutingConfiguration {
@Bean
fun routerFunction(userHandler: MyUserHandler): RouterFunction<ServerResponse> {
return RouterFunctions.route()
.GET("/{user}", ACCEPT_JSON, userHandler::getUser)
.GET("/{user}/customers", ACCEPT_JSON, userHandler::getUserCustomers)
.DELETE("/{user}", ACCEPT_JSON, userHandler::deleteUser)
.build()
}
companion object {
private val ACCEPT_JSON = accept(MediaType.APPLICATION_JSON)
}
257
Java
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.function.ServerRequest;
import org.springframework.web.servlet.function.ServerResponse;
@Component
public class MyUserHandler {
Kotlin
import org.springframework.stereotype.Component
import org.springframework.web.servlet.function.ServerRequest
import org.springframework.web.servlet.function.ServerResponse
@Component
class MyUserHandler {
Spring MVC is part of the core Spring Framework, and detailed information is available in the
258
reference documentation. There are also several guides that cover Spring MVC available at
spring.io/guides.
You can define as many RouterFunction beans as you like to modularize the definition
TIP
of the router. Beans can be ordered if you need to apply a precedence.
Spring Boot provides auto-configuration for Spring MVC that works well with most applications.
• Support for serving static resources, including support for WebJars (covered later in this
document).
If you want to keep those Spring Boot MVC customizations and make more MVC customizations
(interceptors, formatters, view controllers, and other features), you can add your own
@Configuration class of type WebMvcConfigurer but without @EnableWebMvc.
If you want to take complete control of Spring MVC, you can add your own @Configuration
annotated with @EnableWebMvc, or alternatively add your own @Configuration-annotated
DelegatingWebMvcConfiguration as described in the Javadoc of @EnableWebMvc.
Spring MVC uses a different ConversionService to the one used to convert values
from your application.properties or application.yaml file. It means that Period,
Duration and DataSize converters are not available and that @DurationUnit and
@DataSizeUnit annotations will be ignored.
NOTE
If you want to customize the ConversionService used by Spring MVC, you can
provide a WebMvcConfigurer bean with an addFormatters method. From this method
you can register any converter that you like, or you can delegate to the static
methods available on ApplicationConversionService.
HttpMessageConverters
Spring MVC uses the HttpMessageConverter interface to convert HTTP requests and responses.
259
Sensible defaults are included out of the box. For example, objects can be automatically converted
to JSON (by using the Jackson library) or XML (by using the Jackson XML extension, if available, or
by using JAXB if the Jackson XML extension is not available). By default, strings are encoded in UTF-
8.
If you need to add or customize converters, you can use Spring Boot’s HttpMessageConverters class,
as shown in the following listing:
Java
import org.springframework.boot.autoconfigure.http.HttpMessageConverters;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
@Configuration(proxyBeanMethods = false)
public class MyHttpMessageConvertersConfiguration {
@Bean
public HttpMessageConverters customConverters() {
HttpMessageConverter<?> additional = new AdditionalHttpMessageConverter();
HttpMessageConverter<?> another = new AnotherHttpMessageConverter();
return new HttpMessageConverters(additional, another);
}
Kotlin
import org.springframework.boot.autoconfigure.http.HttpMessageConverters
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.http.converter.HttpMessageConverter
@Configuration(proxyBeanMethods = false)
class MyHttpMessageConvertersConfiguration {
@Bean
fun customConverters(): HttpMessageConverters {
val additional: HttpMessageConverter<*> = AdditionalHttpMessageConverter()
val another: HttpMessageConverter<*> = AnotherHttpMessageConverter()
return HttpMessageConverters(additional, another)
}
Any HttpMessageConverter bean that is present in the context is added to the list of converters. You
can also override default converters in the same way.
260
MessageCodesResolver
Spring MVC has a strategy for generating error codes for rendering error messages from binding
errors: MessageCodesResolver. If you set the spring.mvc.message-codes-resolver-format property
PREFIX_ERROR_CODE or POSTFIX_ERROR_CODE, Spring Boot creates one for you (see the enumeration in
DefaultMessageCodesResolver.Format).
Static Content
By default, Spring Boot serves static content from a directory called /static (or /public or
/resources or /META-INF/resources) in the classpath or from the root of the ServletContext. It uses
the ResourceHttpRequestHandler from Spring MVC so that you can modify that behavior by adding
your own WebMvcConfigurer and overriding the addResourceHandlers method.
In a stand-alone web application, the default servlet from the container is not enabled. It can be
enabled using the server.servlet.register-default-servlet property.
The default servlet acts as a fallback, serving content from the root of the ServletContext if Spring
decides not to handle it. Most of the time, this does not happen (unless you modify the default MVC
configuration), because Spring can always handle requests through the DispatcherServlet.
By default, resources are mapped on /**, but you can tune that with the spring.mvc.static-path-
pattern property. For instance, relocating all resources to /resources/** can be achieved as follows:
Properties
spring.mvc.static-path-pattern=/resources/**
Yaml
spring:
mvc:
static-path-pattern: "/resources/**"
You can also customize the static resource locations by using the spring.web.resources.static-
locations property (replacing the default values with a list of directory locations). The root servlet
context path, "/", is automatically added as a location as well.
In addition to the “standard” static resource locations mentioned earlier, a special case is made for
Webjars content. By default, any resources with a path in /webjars/** are served from jar files if
they are packaged in the Webjars format. The path can be customized with the spring.mvc.webjars-
path-pattern property.
Spring Boot also supports the advanced resource handling features provided by Spring MVC,
allowing use cases such as cache-busting static resources or using version agnostic URLs for
261
Webjars.
To use version agnostic URLs for Webjars, add the webjars-locator-core dependency. Then declare
your Webjar. Using jQuery as an example, adding "/webjars/jquery/jquery.min.js" results in
"/webjars/jquery/x.y.z/jquery.min.js" where x.y.z is the Webjar version.
To use cache busting, the following configuration configures a cache busting solution for all static
resources, effectively adding a content hash, such as <link href="/css/spring-
2a2d595e6ed9a0b24f027f2b63b134d6.css"/>, in URLs:
Properties
spring.web.resources.chain.strategy.content.enabled=true
spring.web.resources.chain.strategy.content.paths=/**
Yaml
spring:
web:
resources:
chain:
strategy:
content:
enabled: true
paths: "/**"
When loading resources dynamically with, for example, a JavaScript module loader, renaming files
is not an option. That is why other strategies are also supported and can be combined. A "fixed"
strategy adds a static version string in the URL without changing the file name, as shown in the
following example:
Properties
spring.web.resources.chain.strategy.content.enabled=true
spring.web.resources.chain.strategy.content.paths=/**
spring.web.resources.chain.strategy.fixed.enabled=true
spring.web.resources.chain.strategy.fixed.paths=/js/lib/
spring.web.resources.chain.strategy.fixed.version=v12
262
Yaml
spring:
web:
resources:
chain:
strategy:
content:
enabled: true
paths: "/**"
fixed:
enabled: true
paths: "/js/lib/"
version: "v12"
With this configuration, JavaScript modules located under "/js/lib/" use a fixed versioning
strategy ("/v12/js/lib/mymodule.js"), while other resources still use the content one (<link
href="/css/spring-2a2d595e6ed9a0b24f027f2b63b134d6.css"/>).
This feature has been thoroughly described in a dedicated blog post and in Spring
TIP
Framework’s reference documentation.
Welcome Page
Spring Boot supports both static and templated welcome pages. It first looks for an index.html file in
the configured static content locations. If one is not found, it then looks for an index template. If
either is found, it is automatically used as the welcome page of the application.
Custom Favicon
As with other static resources, Spring Boot checks for a favicon.ico in the configured static content
locations. If such a file is present, it is automatically used as the favicon of the application.
Spring MVC can map incoming HTTP requests to handlers by looking at the request path and
matching it to the mappings defined in your application (for example, @GetMapping annotations on
Controller methods).
Spring Boot chooses to disable suffix pattern matching by default, which means that requests like
"GET /projects/spring-boot.json" will not be matched to @GetMapping("/projects/spring-boot")
mappings. This is considered as a best practice for Spring MVC applications. This feature was
mainly useful in the past for HTTP clients which did not send proper "Accept" request headers; we
needed to make sure to send the correct Content Type to the client. Nowadays, Content Negotiation
is much more reliable.
There are other ways to deal with HTTP clients that do not consistently send proper "Accept"
263
request headers. Instead of using suffix matching, we can use a query parameter to ensure that
requests like "GET /projects/spring-boot?format=json" will be mapped to
@GetMapping("/projects/spring-boot"):
Properties
spring.mvc.contentnegotiation.favor-parameter=true
Yaml
spring:
mvc:
contentnegotiation:
favor-parameter: true
Properties
spring.mvc.contentnegotiation.favor-parameter=true
spring.mvc.contentnegotiation.parameter-name=myparam
Yaml
spring:
mvc:
contentnegotiation:
favor-parameter: true
parameter-name: "myparam"
Most standard media types are supported out-of-the-box, but you can also define new ones:
Properties
spring.mvc.contentnegotiation.media-types.markdown=text/markdown
Yaml
spring:
mvc:
contentnegotiation:
media-types:
markdown: "text/markdown"
As of Spring Framework 5.3, Spring MVC supports several implementation strategies for matching
request paths to Controller handlers. It was previously only supporting the AntPathMatcher strategy,
but it now also offers PathPatternParser. Spring Boot now provides a configuration property to
264
choose and opt in the new strategy:
Properties
spring.mvc.pathmatch.matching-strategy=path-pattern-parser
Yaml
spring:
mvc:
pathmatch:
matching-strategy: "path-pattern-parser"
For more details on why you should consider this new implementation, see the dedicated blog post.
By default, Spring MVC will send a 404 Not Found error response if a handler is not found for a
request. To have a NoHandlerFoundException thrown instead, set configprop:spring.mvc.throw-
exception-if-no-handler-found to true. Note that, by default, the serving of static content is mapped
to /** and will, therefore, provide a handler for all requests. For a NoHandlerFoundException to be
thrown, you must also set spring.mvc.static-path-pattern to a more specific value such as
/resources/** or set spring.web.resources.add-mappings to false to disable serving of static content
entirely.
ConfigurableWebBindingInitializer
Spring MVC uses a WebBindingInitializer to initialize a WebDataBinder for a particular request. If you
create your own ConfigurableWebBindingInitializer @Bean, Spring Boot automatically configures
Spring MVC to use it.
Template Engines
As well as REST web services, you can also use Spring MVC to serve dynamic HTML content. Spring
MVC supports a variety of templating technologies, including Thymeleaf, FreeMarker, and JSPs.
Also, many other templating engines include their own Spring MVC integrations.
Spring Boot includes auto-configuration support for the following templating engines:
• FreeMarker
• Groovy
• Thymeleaf
• Mustache
265
If possible, JSPs should be avoided. There are several known limitations when using
TIP
them with embedded servlet containers.
When you use one of these templating engines with the default configuration, your templates are
picked up automatically from src/main/resources/templates.
Depending on how you run your application, your IDE may order the classpath
differently. Running your application in the IDE from its main method results in a
different ordering than when you run your application by using Maven or Gradle or
TIP
from its packaged jar. This can cause Spring Boot to fail to find the expected template.
If you have this problem, you can reorder the classpath in the IDE to place the
module’s classes and resources first.
Error Handling
By default, Spring Boot provides an /error mapping that handles all errors in a sensible way, and it
is registered as a “global” error page in the servlet container. For machine clients, it produces a
JSON response with details of the error, the HTTP status, and the exception message. For browser
clients, there is a “whitelabel” error view that renders the same data in HTML format (to customize
it, add a View that resolves to error).
There are a number of server.error properties that can be set if you want to customize the default
error handling behavior. See the “Server Properties” section of the Appendix.
To replace the default behavior completely, you can implement ErrorController and register a bean
definition of that type or add a bean of type ErrorAttributes to use the existing mechanism but
replace the contents.
As of Spring Framework 6.0, RFC 7807 Problem Details is supported. Spring MVC can produce
custom error messages with the application/problem+json media type, like:
{
"type": "https://example.org/problems/unknown-project",
"title": "Unknown project",
"status": 404,
"detail": "No project found for id 'spring-unknown'",
"instance": "/projects/spring-unknown"
}
You can also define a class annotated with @ControllerAdvice to customize the JSON document to
266
return for a particular controller and/or exception type, as shown in the following example:
Java
import jakarta.servlet.RequestDispatcher;
import jakarta.servlet.http.HttpServletRequest;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import
org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler;
@ControllerAdvice(basePackageClasses = SomeController.class)
public class MyControllerAdvice extends ResponseEntityExceptionHandler {
@ResponseBody
@ExceptionHandler(MyException.class)
public ResponseEntity<?> handleControllerException(HttpServletRequest request,
Throwable ex) {
HttpStatus status = getStatus(request);
return new ResponseEntity<>(new MyErrorBody(status.value(), ex.getMessage()),
status);
}
267
Kotlin
import jakarta.servlet.RequestDispatcher
import jakarta.servlet.http.HttpServletRequest
import org.springframework.http.HttpStatus
import org.springframework.http.ResponseEntity
import org.springframework.web.bind.annotation.ControllerAdvice
import org.springframework.web.bind.annotation.ExceptionHandler
import org.springframework.web.bind.annotation.ResponseBody
import
org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler
@ControllerAdvice(basePackageClasses = [SomeController::class])
class MyControllerAdvice : ResponseEntityExceptionHandler() {
@ResponseBody
@ExceptionHandler(MyException::class)
fun handleControllerException(request: HttpServletRequest, ex: Throwable):
ResponseEntity<*> {
val status = getStatus(request)
return ResponseEntity(MyErrorBody(status.value(), ex.message), status)
}
In the preceding example, if MyException is thrown by a controller defined in the same package as
SomeController, a JSON representation of the MyErrorBody POJO is used instead of the ErrorAttributes
representation.
In some cases, errors handled at the controller level are not recorded by the metrics infrastructure.
Applications can ensure that such exceptions are recorded with the request metrics by setting the
handled exception as a request attribute:
268
Java
import jakarta.servlet.http.HttpServletRequest;
import org.springframework.boot.web.servlet.error.ErrorAttributes;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ExceptionHandler;
@Controller
public class MyController {
@ExceptionHandler(CustomException.class)
String handleCustomException(HttpServletRequest request, CustomException ex) {
request.setAttribute(ErrorAttributes.ERROR_ATTRIBUTE, ex);
return "errorView";
}
Kotlin
import jakarta.servlet.http.HttpServletRequest
import org.springframework.boot.web.servlet.error.ErrorAttributes
import org.springframework.stereotype.Controller
import org.springframework.web.bind.annotation.ExceptionHandler
@Controller
class MyController {
@ExceptionHandler(CustomException::class)
fun handleCustomException(request: HttpServletRequest, ex: CustomException?):
String {
request.setAttribute(ErrorAttributes.ERROR_ATTRIBUTE, ex)
return "errorView"
}
If you want to display a custom HTML error page for a given status code, you can add a file to an
/error directory. Error pages can either be static HTML (that is, added under any of the static
resource directories) or be built by using templates. The name of the file should be the exact status
code or a series mask.
For example, to map 404 to a static HTML file, your directory structure would be as follows:
269
src/
+- main/
+- java/
| + <source code>
+- resources/
+- public/
+- error/
| +- 404.html
+- <other public assets>
To map all 5xx errors by using a FreeMarker template, your directory structure would be as follows:
src/
+- main/
+- java/
| + <source code>
+- resources/
+- templates/
+- error/
| +- 5xx.ftlh
+- <other templates>
For more complex mappings, you can also add beans that implement the ErrorViewResolver
interface, as shown in the following example:
270
Java
import java.util.Map;
import jakarta.servlet.http.HttpServletRequest;
import org.springframework.boot.autoconfigure.web.servlet.error.ErrorViewResolver;
import org.springframework.http.HttpStatus;
import org.springframework.web.servlet.ModelAndView;
@Override
public ModelAndView resolveErrorView(HttpServletRequest request, HttpStatus
status, Map<String, Object> model) {
// Use the request or status to optionally return a ModelAndView
if (status == HttpStatus.INSUFFICIENT_STORAGE) {
// We could add custom model values here
new ModelAndView("myview");
}
return null;
}
Kotlin
import jakarta.servlet.http.HttpServletRequest
import org.springframework.boot.autoconfigure.web.servlet.error.ErrorViewResolver
import org.springframework.http.HttpStatus
import org.springframework.web.servlet.ModelAndView
You can also use regular Spring MVC features such as @ExceptionHandler methods and
@ControllerAdvice. The ErrorController then picks up any unhandled exceptions.
271
Mapping Error Pages Outside of Spring MVC
For applications that do not use Spring MVC, you can use the ErrorPageRegistrar interface to
directly register ErrorPages. This abstraction works directly with the underlying embedded servlet
container and works even if you do not have a Spring MVC DispatcherServlet.
Java
import org.springframework.boot.web.server.ErrorPage;
import org.springframework.boot.web.server.ErrorPageRegistrar;
import org.springframework.boot.web.server.ErrorPageRegistry;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.HttpStatus;
@Configuration(proxyBeanMethods = false)
public class MyErrorPagesConfiguration {
@Bean
public ErrorPageRegistrar errorPageRegistrar() {
return this::registerErrorPages;
}
272
Kotlin
import org.springframework.boot.web.server.ErrorPage
import org.springframework.boot.web.server.ErrorPageRegistrar
import org.springframework.boot.web.server.ErrorPageRegistry
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.http.HttpStatus
@Configuration(proxyBeanMethods = false)
class MyErrorPagesConfiguration {
@Bean
fun errorPageRegistrar(): ErrorPageRegistrar {
return ErrorPageRegistrar { registry: ErrorPageRegistry ->
registerErrorPages(registry) }
}
If you register an ErrorPage with a path that ends up being handled by a Filter (as is
common with some non-Spring web frameworks, like Jersey and Wicket), then the
NOTE
Filter has to be explicitly registered as an ERROR dispatcher, as shown in the
following example:
273
Java
import java.util.EnumSet;
import jakarta.servlet.DispatcherType;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyFilterConfiguration {
@Bean
public FilterRegistrationBean<MyFilter> myFilter() {
FilterRegistrationBean<MyFilter> registration = new
FilterRegistrationBean<>(new MyFilter());
// ...
registration.setDispatcherTypes(EnumSet.allOf(DispatcherType.class));
return registration;
}
Kotlin
import jakarta.servlet.DispatcherType
import org.springframework.boot.web.servlet.FilterRegistrationBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import java.util.EnumSet
@Configuration(proxyBeanMethods = false)
class MyFilterConfiguration {
@Bean
fun myFilter(): FilterRegistrationBean<MyFilter> {
val registration = FilterRegistrationBean(MyFilter())
// ...
registration.setDispatcherTypes(EnumSet.allOf(DispatcherType::class.java))
return registration
}
Note that the default FilterRegistrationBean does not include the ERROR dispatcher type.
When deployed to a servlet container, Spring Boot uses its error page filter to forward a request
274
with an error status to the appropriate error page. This is necessary as the servlet specification does
not provide an API for registering error pages. Depending on the container that you are deploying
your war file to and the technologies that your application uses, some additional configuration may
be required.
The error page filter can only forward the request to the correct error page if the response has not
already been committed. By default, WebSphere Application Server 8.0 and later commits the
response upon successful completion of a servlet’s service method. You should disable this behavior
by setting com.ibm.ws.webcontainer.invokeFlushAfterService to false.
CORS Support
Cross-origin resource sharing (CORS) is a W3C specification implemented by most browsers that
lets you specify in a flexible way what kind of cross-domain requests are authorized, instead of
using some less secure and less powerful approaches such as IFRAME or JSONP.
As of version 4.2, Spring MVC supports CORS. Using controller method CORS configuration with
@CrossOrigin annotations in your Spring Boot application does not require any specific
configuration. Global CORS configuration can be defined by registering a WebMvcConfigurer bean
with a customized addCorsMappings(CorsRegistry) method, as shown in the following example:
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration(proxyBeanMethods = false)
public class MyCorsConfiguration {
@Bean
public WebMvcConfigurer corsConfigurer() {
return new WebMvcConfigurer() {
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/api/**");
}
};
}
275
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.web.servlet.config.annotation.CorsRegistry
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer
@Configuration(proxyBeanMethods = false)
class MyCorsConfiguration {
@Bean
fun corsConfigurer(): WebMvcConfigurer {
return object : WebMvcConfigurer {
override fun addCorsMappings(registry: CorsRegistry) {
registry.addMapping("/api/**")
}
}
}
If you prefer the JAX-RS programming model for REST endpoints, you can use one of the available
implementations instead of Spring MVC. Jersey and Apache CXF work quite well out of the box. CXF
requires you to register its Servlet or Filter as a @Bean in your application context. Jersey has some
native Spring support, so we also provide auto-configuration support for it in Spring Boot, together
with a starter.
To get started with Jersey, include the spring-boot-starter-jersey as a dependency and then you
need one @Bean of type ResourceConfig in which you register all the endpoints, as shown in the
following example:
import org.glassfish.jersey.server.ResourceConfig;
import org.springframework.stereotype.Component;
@Component
public class MyJerseyConfig extends ResourceConfig {
public MyJerseyConfig() {
register(MyEndpoint.class);
}
276
Jersey’s support for scanning executable archives is rather limited. For
example, it cannot scan for endpoints in a package found in a fully executable
jar file or in WEB-INF/classes when running an executable war file. To avoid
WARNING
this limitation, the packages method should not be used, and endpoints should
be registered individually by using the register method, as shown in the
preceding example.
For more advanced customizations, you can also register an arbitrary number of beans that
implement ResourceConfigCustomizer.
All the registered endpoints should be @Components with HTTP resource annotations (@GET and
others), as shown in the following example:
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import org.springframework.stereotype.Component;
@Component
@Path("/hello")
public class MyEndpoint {
@GET
public String message() {
return "Hello";
}
Since the Endpoint is a Spring @Component, its lifecycle is managed by Spring and you can use the
@Autowired annotation to inject dependencies and use the @Value annotation to inject external
configuration. By default, the Jersey servlet is registered and mapped to /*. You can change the
mapping by adding @ApplicationPath to your ResourceConfig.
277
8.1.3. Embedded Servlet Container Support
For servlet application, Spring Boot includes support for embedded Tomcat, Jetty, and Undertow
servers. Most developers use the appropriate “Starter” to obtain a fully configured instance. By
default, the embedded server listens for HTTP requests on port 8080.
When using an embedded servlet container, you can register servlets, filters, and all the listeners
(such as HttpSessionListener) from the servlet spec, either by using Spring beans or by scanning for
servlet components.
Any Servlet, Filter, or servlet *Listener instance that is a Spring bean is registered with the
embedded container. This can be particularly convenient if you want to refer to a value from your
application.properties during configuration.
By default, if the context contains only a single Servlet, it is mapped to /. In the case of multiple
servlet beans, the bean name is used as a path prefix. Filters map to /*.
If convention-based mapping is not flexible enough, you can use the ServletRegistrationBean,
FilterRegistrationBean, and ServletListenerRegistrationBean classes for complete control.
It is usually safe to leave filter beans unordered. If a specific order is required, you should annotate
the Filter with @Order or make it implement Ordered. You cannot configure the order of a Filter by
annotating its bean method with @Order. If you cannot change the Filter class to add @Order or
implement Ordered, you must define a FilterRegistrationBean for the Filter and set the registration
bean’s order using the setOrder(int) method. Avoid configuring a filter that reads the request body
at Ordered.HIGHEST_PRECEDENCE, since it might go against the character encoding configuration of
your application. If a servlet filter wraps the request, it should be configured with an order that is
less than or equal to OrderedFilter.REQUEST_WRAPPER_FILTER_MAX_ORDER.
To see the order of every Filter in your application, enable debug level logging for the
TIP web logging group (logging.level.web=debug). Details of the registered filters, including
their order and URL patterns, will then be logged at startup.
Take care when registering Filter beans since they are initialized very early in
WARNING the application lifecycle. If you need to register a Filter that interacts with
other beans, consider using a DelegatingFilterProxyRegistrationBean instead.
278
If you need to perform servlet context initialization in a Spring Boot application, you should
register a bean that implements the
org.springframework.boot.web.servlet.ServletContextInitializer interface. The single onStartup
method provides access to the ServletContext and, if necessary, can easily be used as an adapter to
an existing WebApplicationInitializer.
When using an embedded container, automatic registration of classes annotated with @WebServlet,
@WebFilter, and @WebListener can be enabled by using @ServletComponentScan.
The ServletWebServerApplicationContext
Under the hood, Spring Boot uses a different type of ApplicationContext for embedded servlet
container support. The ServletWebServerApplicationContext is a special type of
WebApplicationContext that bootstraps itself by searching for a single ServletWebServerFactory bean.
Usually a TomcatServletWebServerFactory, JettyServletWebServerFactory, or
UndertowServletWebServerFactory has been auto-configured.
In an embedded container setup, the ServletContext is set as part of server startup which happens
during application context initialization. Because of this beans in the ApplicationContext cannot be
reliably initialized with a ServletContext. One way to get around this is to inject ApplicationContext
as a dependency of the bean and access the ServletContext only when it is needed. Another way is
to use a callback once the server has started. This can be done using an ApplicationListener which
listens for the ApplicationStartedEvent as follows:
279
import jakarta.servlet.ServletContext;
import org.springframework.boot.context.event.ApplicationStartedEvent;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationListener;
import org.springframework.web.context.WebApplicationContext;
@Override
public void onApplicationEvent(ApplicationStartedEvent event) {
ApplicationContext applicationContext = event.getApplicationContext();
this.servletContext = ((WebApplicationContext)
applicationContext).getServletContext();
}
Common servlet container settings can be configured by using Spring Environment properties.
Usually, you would define the properties in your application.properties or application.yaml file.
• Network settings: Listen port for incoming HTTP requests (server.port), interface address to
bind to server.address, and so on.
• SSL
• HTTP compression
Spring Boot tries as much as possible to expose common settings, but this is not always possible. For
those cases, dedicated namespaces offer server-specific customizations (see server.tomcat and
server.undertow). For instance, access logs can be configured with specific features of the embedded
servlet container.
SameSite Cookies
The SameSite cookie attribute can be used by web browsers to control if and how cookies are
280
submitted in cross-site requests. The attribute is particularly relevant for modern web browsers
which have started to change the default value that is used when the attribute is missing.
If you want to change the SameSite attribute of your session cookie, you can use the
server.servlet.session.cookie.same-site property. This property is supported by auto-configured
Tomcat, Jetty and Undertow servers. It is also used to configure Spring Session servlet based
SessionRepository beans.
For example, if you want your session cookie to have a SameSite attribute of None, you can add the
following to your application.properties or application.yaml file:
Properties
server.servlet.session.cookie.same-site=none
Yaml
server:
servlet:
session:
cookie:
same-site: "none"
If you want to change the SameSite attribute on other cookies added to your HttpServletResponse,
you can use a CookieSameSiteSupplier. The CookieSameSiteSupplier is passed a Cookie and may
return a SameSite value, or null.
There are a number of convenience factory and filter methods that you can use to quickly match
specific cookies. For example, adding the following bean will automatically apply a SameSite of Lax
for all cookies with a name that matches the regular expression myapp.*.
Java
import org.springframework.boot.web.servlet.server.CookieSameSiteSupplier;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MySameSiteConfiguration {
@Bean
public CookieSameSiteSupplier applicationCookieSameSiteSupplier() {
return CookieSameSiteSupplier.ofLax().whenHasNameMatching("myapp.*");
}
281
Kotlin
import org.springframework.boot.web.servlet.server.CookieSameSiteSupplier
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MySameSiteConfiguration {
@Bean
fun applicationCookieSameSiteSupplier(): CookieSameSiteSupplier {
return CookieSameSiteSupplier.ofLax().whenHasNameMatching("myapp.*")
}
Programmatic Customization
If you need to programmatically configure your embedded servlet container, you can register a
Spring bean that implements the WebServerFactoryCustomizer interface. WebServerFactoryCustomizer
provides access to the ConfigurableServletWebServerFactory, which includes numerous
customization setter methods. The following example shows programmatically setting the port:
Java
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import
org.springframework.boot.web.servlet.server.ConfigurableServletWebServerFactory;
import org.springframework.stereotype.Component;
@Component
public class MyWebServerFactoryCustomizer implements
WebServerFactoryCustomizer<ConfigurableServletWebServerFactory> {
@Override
public void customize(ConfigurableServletWebServerFactory server) {
server.setPort(9000);
}
282
Kotlin
import org.springframework.boot.web.server.WebServerFactoryCustomizer
import org.springframework.boot.web.servlet.server.ConfigurableServletWebServerFactory
import org.springframework.stereotype.Component
@Component
class MyWebServerFactoryCustomizer :
WebServerFactoryCustomizer<ConfigurableServletWebServerFactory> {
Java
import java.time.Duration;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.stereotype.Component;
@Component
public class MyTomcatWebServerFactoryCustomizer implements
WebServerFactoryCustomizer<TomcatServletWebServerFactory> {
@Override
public void customize(TomcatServletWebServerFactory server) {
server.addConnectorCustomizers((connector) ->
connector.setAsyncTimeout(Duration.ofSeconds(20).toMillis()));
}
283
Kotlin
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory
import org.springframework.boot.web.server.WebServerFactoryCustomizer
import org.springframework.stereotype.Component
import java.time.Duration
@Component
class MyTomcatWebServerFactoryCustomizer :
WebServerFactoryCustomizer<TomcatServletWebServerFactory> {
For more advanced use cases that require you to extend from ServletWebServerFactory, you can
expose a bean of such type yourself.
Setters are provided for many configuration options. Several protected method “hooks” are also
provided should you need to do something more exotic. See the source code documentation for
details.
Auto-configured customizers are still applied on your custom factory, so use that
NOTE
option carefully.
JSP Limitations
When running a Spring Boot application that uses an embedded servlet container (and is packaged
as an executable archive), there are some limitations in the JSP support.
• With Jetty and Tomcat, it should work if you use war packaging. An executable war will work
when launched with java -jar, and will also be deployable to any standard container. JSPs are
not supported when using an executable jar.
• Creating a custom error.jsp page does not override the default view for error handling. Custom
error pages should be used instead.
284
8.2.1. The “Spring WebFlux Framework”
Spring WebFlux is the new reactive web framework introduced in Spring Framework 5.0. Unlike
Spring MVC, it does not require the servlet API, is fully asynchronous and non-blocking, and
implements the Reactive Streams specification through the Reactor project.
Spring WebFlux comes in two flavors: functional and annotation-based. The annotation-based one
is quite close to the Spring MVC model, as shown in the following example:
285
Java
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/users")
public class MyRestController {
@GetMapping("/{userId}")
public Mono<User> getUser(@PathVariable Long userId) {
return this.userRepository.findById(userId);
}
@GetMapping("/{userId}/customers")
public Flux<Customer> getUserCustomers(@PathVariable Long userId) {
return
this.userRepository.findById(userId).flatMapMany(this.customerRepository::findByUser);
}
@DeleteMapping("/{userId}")
public Mono<Void> deleteUser(@PathVariable Long userId) {
return this.userRepository.deleteById(userId);
}
286
Kotlin
import org.springframework.web.bind.annotation.DeleteMapping
import org.springframework.web.bind.annotation.GetMapping
import org.springframework.web.bind.annotation.PathVariable
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono
@RestController
@RequestMapping("/users")
class MyRestController(private val userRepository: UserRepository, private val
customerRepository: CustomerRepository) {
@GetMapping("/{userId}")
fun getUser(@PathVariable userId: Long): Mono<User?> {
return userRepository.findById(userId)
}
@GetMapping("/{userId}/customers")
fun getUserCustomers(@PathVariable userId: Long): Flux<Customer> {
return userRepository.findById(userId).flatMapMany { user: User? ->
customerRepository.findByUser(user)
}
}
@DeleteMapping("/{userId}")
fun deleteUser(@PathVariable userId: Long): Mono<Void> {
return userRepository.deleteById(userId)
}
“WebFlux.fn”, the functional variant, separates the routing configuration from the actual handling
of the requests, as shown in the following example:
287
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.server.RequestPredicate;
import org.springframework.web.reactive.function.server.RouterFunction;
import org.springframework.web.reactive.function.server.ServerResponse;
import static
org.springframework.web.reactive.function.server.RequestPredicates.accept;
import static org.springframework.web.reactive.function.server.RouterFunctions.route;
@Configuration(proxyBeanMethods = false)
public class MyRoutingConfiguration {
@Bean
public RouterFunction<ServerResponse> monoRouterFunction(MyUserHandler
userHandler) {
return route()
.GET("/{user}", ACCEPT_JSON, userHandler::getUser)
.GET("/{user}/customers", ACCEPT_JSON, userHandler::getUserCustomers)
.DELETE("/{user}", ACCEPT_JSON, userHandler::deleteUser)
.build();
}
288
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.http.MediaType
import org.springframework.web.reactive.function.server.RequestPredicates.DELETE
import org.springframework.web.reactive.function.server.RequestPredicates.GET
import org.springframework.web.reactive.function.server.RequestPredicates.accept
import org.springframework.web.reactive.function.server.RouterFunction
import org.springframework.web.reactive.function.server.RouterFunctions
import org.springframework.web.reactive.function.server.ServerResponse
@Configuration(proxyBeanMethods = false)
class MyRoutingConfiguration {
@Bean
fun monoRouterFunction(userHandler: MyUserHandler): RouterFunction<ServerResponse>
{
return RouterFunctions.route(
GET("/{user}").and(ACCEPT_JSON), userHandler::getUser).andRoute(
GET("/{user}/customers").and(ACCEPT_JSON),
userHandler::getUserCustomers).andRoute(
DELETE("/{user}").and(ACCEPT_JSON), userHandler::deleteUser)
}
companion object {
private val ACCEPT_JSON = accept(MediaType.APPLICATION_JSON)
}
289
Java
import reactor.core.publisher.Mono;
import org.springframework.stereotype.Component;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.reactive.function.server.ServerResponse;
@Component
public class MyUserHandler {
Kotlin
import org.springframework.stereotype.Component
import org.springframework.web.reactive.function.server.ServerRequest
import org.springframework.web.reactive.function.server.ServerResponse
import reactor.core.publisher.Mono
@Component
class MyUserHandler {
WebFlux is part of the Spring Framework and detailed information is available in its reference
290
documentation.
You can define as many RouterFunction beans as you like to modularize the definition
TIP
of the router. Beans can be ordered if you need to apply a precedence.
Spring Boot provides auto-configuration for Spring WebFlux that works well with most
applications.
• Support for serving static resources, including support for WebJars (described later in this
document).
If you want to keep Spring Boot WebFlux features and you want to add additional WebFlux
configuration, you can add your own @Configuration class of type WebFluxConfigurer but without
@EnableWebFlux.
If you want to take complete control of Spring WebFlux, you can add your own @Configuration
annotated with @EnableWebFlux.
Spring WebFlux uses the HttpMessageReader and HttpMessageWriter interfaces to convert HTTP
requests and responses. They are configured with CodecConfigurer to have sensible defaults by
looking at the libraries available in your classpath.
Spring Boot provides dedicated configuration properties for codecs, spring.codec.*. It also applies
further customization by using CodecCustomizer instances. For example, spring.jackson.*
configuration keys are applied to the Jackson codec.
If you need to add or customize codecs, you can create a custom CodecCustomizer component, as
shown in the following example:
291
Java
import org.springframework.boot.web.codec.CodecCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.codec.ServerSentEventHttpMessageReader;
@Configuration(proxyBeanMethods = false)
public class MyCodecsConfiguration {
@Bean
public CodecCustomizer myCodecCustomizer() {
return (configurer) -> {
configurer.registerDefaults(false);
configurer.customCodecs().register(new
ServerSentEventHttpMessageReader());
// ...
};
}
Kotlin
import org.springframework.boot.web.codec.CodecCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.http.codec.CodecConfigurer
import org.springframework.http.codec.ServerSentEventHttpMessageReader
class MyCodecsConfiguration {
@Bean
fun myCodecCustomizer(): CodecCustomizer {
return CodecCustomizer { configurer: CodecConfigurer ->
configurer.registerDefaults(false)
configurer.customCodecs().register(ServerSentEventHttpMessageReader())
}
}
You can also leverage Boot’s custom JSON serializers and deserializers.
Static Content
By default, Spring Boot serves static content from a directory called /static (or /public or
/resources or /META-INF/resources) in the classpath. It uses the ResourceWebHandler from Spring
WebFlux so that you can modify that behavior by adding your own WebFluxConfigurer and
overriding the addResourceHandlers method.
292
By default, resources are mapped on /**, but you can tune that by setting the
spring.webflux.static-path-pattern property. For instance, relocating all resources to /resources/**
can be achieved as follows:
Properties
spring.webflux.static-path-pattern=/resources/**
Yaml
spring:
webflux:
static-path-pattern: "/resources/**"
You can also customize the static resource locations by using spring.web.resources.static-
locations. Doing so replaces the default values with a list of directory locations. If you do so, the
default welcome page detection switches to your custom locations. So, if there is an index.html in
any of your locations on startup, it is the home page of the application.
In addition to the “standard” static resource locations listed earlier, a special case is made for
Webjars content. By default, any resources with a path in /webjars/** are served from jar files if
they are packaged in the Webjars format. The path can be customized with the
spring.webflux.webjars-path-pattern property.
Spring WebFlux applications do not strictly depend on the servlet API, so they cannot
TIP
be deployed as war files and do not use the src/main/webapp directory.
Welcome Page
Spring Boot supports both static and templated welcome pages. It first looks for an index.html file in
the configured static content locations. If one is not found, it then looks for an index template. If
either is found, it is automatically used as the welcome page of the application.
Template Engines
As well as REST web services, you can also use Spring WebFlux to serve dynamic HTML content.
Spring WebFlux supports a variety of templating technologies, including Thymeleaf, FreeMarker,
and Mustache.
Spring Boot includes auto-configuration support for the following templating engines:
• FreeMarker
• Thymeleaf
• Mustache
When you use one of these templating engines with the default configuration, your templates are
picked up automatically from src/main/resources/templates.
293
Error Handling
Spring Boot provides a WebExceptionHandler that handles all errors in a sensible way. Its position in
the processing order is immediately before the handlers provided by WebFlux, which are
considered last. For machine clients, it produces a JSON response with details of the error, the HTTP
status, and the exception message. For browser clients, there is a “whitelabel” error handler that
renders the same data in HTML format. You can also provide your own HTML templates to display
errors (see the next section).
Before customizing error handling in Spring Boot directly, you can leverage the RFC 7807 Problem
Details support in Spring WebFlux. Spring WebFlux can produce custom error messages with the
application/problem+json media type, like:
{
"type": "https://example.org/problems/unknown-project",
"title": "Unknown project",
"status": 404,
"detail": "No project found for id 'spring-unknown'",
"instance": "/projects/spring-unknown"
}
The first step to customizing this feature often involves using the existing mechanism but replacing
or augmenting the error contents. For that, you can add a bean of type ErrorAttributes.
To change the error handling behavior, you can implement ErrorWebExceptionHandler and register a
bean definition of that type. Because an ErrorWebExceptionHandler is quite low-level, Spring Boot
also provides a convenient AbstractErrorWebExceptionHandler to let you handle errors in a WebFlux
functional way, as shown in the following example:
294
Java
import reactor.core.publisher.Mono;
import org.springframework.boot.autoconfigure.web.WebProperties.Resources;
import
org.springframework.boot.autoconfigure.web.reactive.error.AbstractErrorWebExceptionHan
dler;
import org.springframework.boot.web.reactive.error.ErrorAttributes;
import org.springframework.context.ApplicationContext;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import org.springframework.web.reactive.function.server.RouterFunction;
import org.springframework.web.reactive.function.server.RouterFunctions;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.reactive.function.server.ServerResponse;
import org.springframework.web.reactive.function.server.ServerResponse.BodyBuilder;
@Component
public class MyErrorWebExceptionHandler extends AbstractErrorWebExceptionHandler {
@Override
protected RouterFunction<ServerResponse> getRoutingFunction(ErrorAttributes
errorAttributes) {
return RouterFunctions.route(this::acceptsXml, this::handleErrorAsXml);
}
295
Kotlin
import org.springframework.boot.autoconfigure.web.WebProperties
import
org.springframework.boot.autoconfigure.web.reactive.error.AbstractErrorWebExceptionHan
dler
import org.springframework.boot.web.reactive.error.ErrorAttributes
import org.springframework.context.ApplicationContext
import org.springframework.http.HttpStatus
import org.springframework.http.MediaType
import org.springframework.stereotype.Component
import org.springframework.web.reactive.function.server.RouterFunction
import org.springframework.web.reactive.function.server.RouterFunctions
import org.springframework.web.reactive.function.server.ServerRequest
import org.springframework.web.reactive.function.server.ServerResponse
import reactor.core.publisher.Mono
@Component
class MyErrorWebExceptionHandler(errorAttributes: ErrorAttributes?, resources:
WebProperties.Resources?,
applicationContext: ApplicationContext?) :
AbstractErrorWebExceptionHandler(errorAttributes, resources, applicationContext) {
For a more complete picture, you can also subclass DefaultErrorWebExceptionHandler directly and
override specific methods.
In some cases, errors handled at the controller or handler function level are not recorded by the
metrics infrastructure. Applications can ensure that such exceptions are recorded with the request
metrics by setting the handled exception as a request attribute:
296
Java
import org.springframework.boot.web.reactive.error.ErrorAttributes;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.reactive.result.view.Rendering;
import org.springframework.web.server.ServerWebExchange;
@Controller
public class MyExceptionHandlingController {
@GetMapping("/profile")
public Rendering userProfile() {
// ...
throw new IllegalStateException();
}
@ExceptionHandler(IllegalStateException.class)
public Rendering handleIllegalState(ServerWebExchange exchange,
IllegalStateException exc) {
exchange.getAttributes().putIfAbsent(ErrorAttributes.ERROR_ATTRIBUTE, exc);
return Rendering.view("errorView").modelAttribute("message",
exc.getMessage()).build();
}
297
Kotlin
import org.springframework.boot.web.reactive.error.ErrorAttributes
import org.springframework.stereotype.Controller
import org.springframework.web.bind.annotation.ExceptionHandler
import org.springframework.web.bind.annotation.GetMapping
import org.springframework.web.reactive.result.view.Rendering
import org.springframework.web.server.ServerWebExchange
@Controller
class MyExceptionHandlingController {
@GetMapping("/profile")
fun userProfile(): Rendering {
// ...
throw IllegalStateException()
}
@ExceptionHandler(IllegalStateException::class)
fun handleIllegalState(exchange: ServerWebExchange, exc: IllegalStateException):
Rendering {
exchange.attributes.putIfAbsent(ErrorAttributes.ERROR_ATTRIBUTE, exc)
return Rendering.view("errorView").modelAttribute("message", exc.message ?:
"").build()
}
If you want to display a custom HTML error page for a given status code, you can add views that
resolve from error/*, for example by adding files to a /error directory. Error pages can either be
static HTML (that is, added under any of the static resource directories) or built with templates. The
name of the file should be the exact status code, a status code series mask, or error for a default if
nothing else matches. Note that the path to the default error view is error/error, whereas with
Spring MVC the default error view is error.
For example, to map 404 to a static HTML file, your directory structure would be as follows:
src/
+- main/
+- java/
| + <source code>
+- resources/
+- public/
+- error/
| +- 404.html
+- <other public assets>
298
To map all 5xx errors by using a Mustache template, your directory structure would be as follows:
src/
+- main/
+- java/
| + <source code>
+- resources/
+- templates/
+- error/
| +- 5xx.mustache
+- <other templates>
Web Filters
Spring WebFlux provides a WebFilter interface that can be implemented to filter HTTP request-
response exchanges. WebFilter beans found in the application context will be automatically used to
filter each exchange.
Where the order of the filters is important they can implement Ordered or be annotated with @Order.
Spring Boot auto-configuration may configure web filters for you. When it does so, the orders
shown in the following table will be used:
HttpExchangesWebFilter Ordered.LOWEST_PRECEDENCE - 10
Spring Boot includes support for the following embedded reactive web servers: Reactor Netty,
Tomcat, Jetty, and Undertow. Most developers use the appropriate “Starter” to obtain a fully
configured instance. By default, the embedded server listens for HTTP requests on port 8080.
When auto-configuring a Reactor Netty or Jetty server, Spring Boot will create specific beans that
will provide HTTP resources to the server instance: ReactorResourceFactory or JettyResourceFactory.
By default, those resources will be also shared with the Reactor Netty and Jetty clients for optimal
performances, given:
• the client instance is built using the WebClient.Builder bean auto-configured by Spring Boot
Developers can override the resource configuration for Jetty and Reactor Netty by providing a
custom ReactorResourceFactory or JettyResourceFactory bean - this will be applied to both clients
299
and servers.
You can learn more about the resource configuration on the client side in the WebClient Runtime
section.
To enable graceful shutdown, configure the server.shutdown property, as shown in the following
example:
Properties
server.shutdown=graceful
Yaml
server:
shutdown: "graceful"
Properties
spring.lifecycle.timeout-per-shutdown-phase=20s
Yaml
spring:
lifecycle:
timeout-per-shutdown-phase: "20s"
Using graceful shutdown with your IDE may not work properly if it does not
IMPORTANT send a proper SIGTERM signal. See the documentation of your IDE for more
details.
300
8.4. Spring Security
If Spring Security is on the classpath, then web applications are secured by default. Spring Boot
relies on Spring Security’s content-negotiation strategy to determine whether to use httpBasic or
formLogin. To add method-level security to a web application, you can also add
@EnableGlobalMethodSecurity with your desired settings. Additional information can be found in the
Spring Security Reference Guide.
The default UserDetailsService has a single user. The user name is user, and the password is
random and is printed at WARN level when the application starts, as shown in the following
example:
This generated password is for development use only. Your security configuration must
be updated before running your application in production.
You can change the username and password by providing a spring.security.user.name and
spring.security.user.password.
• Form-based login or HTTP Basic security (depending on the Accept header in the request) for the
entire application (including actuator endpoints if actuator is on the classpath).
To also switch off the UserDetailsService configuration, you can add a bean of type
UserDetailsService, AuthenticationProvider, or AuthenticationManager.
301
Access rules can be overridden by adding a custom SecurityFilterChain bean. Spring Boot provides
convenience methods that can be used to override access rules for actuator endpoints and static
resources. EndpointRequest can be used to create a RequestMatcher that is based on the
management.endpoints.web.base-path property. PathRequest can be used to create a RequestMatcher for
resources in commonly used locations.
Similar to Spring MVC applications, you can secure your WebFlux applications by adding the
spring-boot-starter-security dependency. The default security configuration is implemented in
ReactiveSecurityAutoConfiguration and UserDetailsServiceAutoConfiguration.
ReactiveSecurityAutoConfiguration imports WebFluxSecurityConfiguration for web security and
UserDetailsServiceAutoConfiguration configures authentication, which is also relevant in non-web
applications. To switch off the default web application security configuration completely, you can
add a bean of type WebFilterChainProxy (doing so does not disable the UserDetailsService
configuration or Actuator’s security).
To also switch off the UserDetailsService configuration, you can add a bean of type
ReactiveUserDetailsService or ReactiveAuthenticationManager.
Access rules and the use of multiple Spring Security components such as OAuth 2 Client and
Resource Server can be configured by adding a custom SecurityWebFilterChain bean. Spring Boot
provides convenience methods that can be used to override access rules for actuator endpoints and
static resources. EndpointRequest can be used to create a ServerWebExchangeMatcher that is based on
the management.endpoints.web.base-path property.
For example, you can customize your security configuration by adding something like:
302
Java
import org.springframework.boot.autoconfigure.security.reactive.PathRequest;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.web.server.ServerHttpSecurity;
import org.springframework.security.web.server.SecurityWebFilterChain;
@Configuration(proxyBeanMethods = false)
public class MyWebFluxSecurityConfiguration {
@Bean
public SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) {
http.authorizeExchange((exchange) -> {
exchange.matchers(PathRequest.toStaticResources().atCommonLocations()).permitAll();
exchange.pathMatchers("/foo", "/bar").authenticated();
});
http.formLogin(withDefaults());
return http.build();
}
303
Kotlin
import org.springframework.boot.autoconfigure.security.reactive.PathRequest
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.security.config.Customizer
import org.springframework.security.config.Customizer.withDefaults
import org.springframework.security.config.web.server.ServerHttpSecurity
import org.springframework.security.web.server.SecurityWebFilterChain
@Configuration(proxyBeanMethods = false)
class MyWebFluxSecurityConfiguration {
@Bean
fun springSecurityFilterChain(http: ServerHttpSecurity): SecurityWebFilterChain {
http.authorizeExchange { spec ->
spec.matchers(PathRequest.toStaticResources().atCommonLocations()).permitAll()
spec.pathMatchers("/foo", "/bar").authenticated()
}
http.formLogin(withDefaults())
return http.build()
}
8.4.3. OAuth2
Client
If you have spring-security-oauth2-client on your classpath, you can take advantage of some auto-
configuration to set up OAuth2/Open ID Connect clients. This configuration makes use of the
properties under OAuth2ClientProperties. The same properties are applicable to both servlet and
reactive applications.
You can register multiple OAuth2 clients and providers under the spring.security.oauth2.client
prefix, as shown in the following example:
304
Properties
spring.security.oauth2.client.registration.my-client-1.client-id=abcd
spring.security.oauth2.client.registration.my-client-1.client-secret=password
spring.security.oauth2.client.registration.my-client-1.client-name=Client for user
scope
spring.security.oauth2.client.registration.my-client-1.provider=my-oauth-provider
spring.security.oauth2.client.registration.my-client-1.scope=user
spring.security.oauth2.client.registration.my-client-1.redirect-uri=https://my-
redirect-uri.com
spring.security.oauth2.client.registration.my-client-1.client-authentication-
method=basic
spring.security.oauth2.client.registration.my-client-1.authorization-grant-
type=authorization_code
spring.security.oauth2.client.registration.my-client-2.client-id=abcd
spring.security.oauth2.client.registration.my-client-2.client-secret=password
spring.security.oauth2.client.registration.my-client-2.client-name=Client for email
scope
spring.security.oauth2.client.registration.my-client-2.provider=my-oauth-provider
spring.security.oauth2.client.registration.my-client-2.scope=email
spring.security.oauth2.client.registration.my-client-2.redirect-uri=https://my-
redirect-uri.com
spring.security.oauth2.client.registration.my-client-2.client-authentication-
method=basic
spring.security.oauth2.client.registration.my-client-2.authorization-grant-
type=authorization_code
spring.security.oauth2.client.provider.my-oauth-provider.authorization-uri=https://my-
auth-server/oauth/authorize
spring.security.oauth2.client.provider.my-oauth-provider.token-uri=https://my-auth-
server/oauth/token
spring.security.oauth2.client.provider.my-oauth-provider.user-info-uri=https://my-
auth-server/userinfo
spring.security.oauth2.client.provider.my-oauth-provider.user-info-authentication-
method=header
spring.security.oauth2.client.provider.my-oauth-provider.jwk-set-uri=https://my-auth-
server/token_keys
spring.security.oauth2.client.provider.my-oauth-provider.user-name-attribute=name
305
Yaml
spring:
security:
oauth2:
client:
registration:
my-client-1:
client-id: "abcd"
client-secret: "password"
client-name: "Client for user scope"
provider: "my-oauth-provider"
scope: "user"
redirect-uri: "https://my-redirect-uri.com"
client-authentication-method: "basic"
authorization-grant-type: "authorization_code"
my-client-2:
client-id: "abcd"
client-secret: "password"
client-name: "Client for email scope"
provider: "my-oauth-provider"
scope: "email"
redirect-uri: "https://my-redirect-uri.com"
client-authentication-method: "basic"
authorization-grant-type: "authorization_code"
provider:
my-oauth-provider:
authorization-uri: "https://my-auth-server/oauth/authorize"
token-uri: "https://my-auth-server/oauth/token"
user-info-uri: "https://my-auth-server/userinfo"
user-info-authentication-method: "header"
jwk-set-uri: "https://my-auth-server/token_keys"
user-name-attribute: "name"
For OpenID Connect providers that support OpenID Connect discovery, the configuration can be
further simplified. The provider needs to be configured with an issuer-uri which is the URI that it
asserts as its Issuer Identifier. For example, if the issuer-uri provided is "https://example.com",
then an "OpenID Provider Configuration Request" will be made to "https://example.com/.well-
known/openid-configuration". The result is expected to be an "OpenID Provider Configuration
Response". The following example shows how an OpenID Connect Provider can be configured with
the issuer-uri:
Properties
spring.security.oauth2.client.provider.oidc-provider.issuer-uri=https://dev-
123456.oktapreview.com/oauth2/default/
306
Yaml
spring:
security:
oauth2:
client:
provider:
oidc-provider:
issuer-uri: "https://dev-123456.oktapreview.com/oauth2/default/"
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MyOAuthClientConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.authorizeHttpRequests((requests) ->
requests.anyRequest().authenticated());
http.oauth2Login((login) -> login.redirectionEndpoint((endpoint) ->
endpoint.baseUri("custom-callback")));
return http.build();
}
307
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.security.config.annotation.web.builders.HttpSecurity
import org.springframework.security.web.SecurityFilterChain
@Configuration(proxyBeanMethods = false)
class MyOAuthClientConfiguration {
@Bean
fun securityFilterChain(http: HttpSecurity): SecurityFilterChain {
http.authorizeHttpRequests { requests -> requests.anyRequest().authenticated()
}
http.oauth2Login { login -> login.redirectionEndpoint { redirectionEndpoint ->
redirectionEndpoint.baseUri("custom-callback") } }
return http.build()
}
For common OAuth2 and OpenID providers, including Google, Github, Facebook, and Okta, we
provide a set of provider defaults (google, github, facebook, and okta, respectively).
If you do not need to customize these providers, you can set the provider attribute to the one for
which you need to infer defaults. Also, if the key for the client registration matches a default
supported provider, Spring Boot infers that as well.
In other words, the two configurations in the following example use the Google provider:
Properties
spring.security.oauth2.client.registration.my-client.client-id=abcd
spring.security.oauth2.client.registration.my-client.client-secret=password
spring.security.oauth2.client.registration.my-client.provider=google
spring.security.oauth2.client.registration.google.client-id=abcd
spring.security.oauth2.client.registration.google.client-secret=password
308
Yaml
spring:
security:
oauth2:
client:
registration:
my-client:
client-id: "abcd"
client-secret: "password"
provider: "google"
google:
client-id: "abcd"
client-secret: "password"
Resource Server
Properties
spring.security.oauth2.resourceserver.jwt.jwk-set-
uri=https://example.com/oauth2/default/v1/keys
Yaml
spring:
security:
oauth2:
resourceserver:
jwt:
jwk-set-uri: "https://example.com/oauth2/default/v1/keys"
Properties
spring.security.oauth2.resourceserver.jwt.issuer-uri=https://dev-
123456.oktapreview.com/oauth2/default/
Yaml
spring:
security:
oauth2:
resourceserver:
jwt:
issuer-uri: "https://dev-123456.oktapreview.com/oauth2/default/"
309
If the authorization server does not support a JWK Set URI, you can configure the
resource server with the Public Key used for verifying the signature of the JWT. This
NOTE can be done using the spring.security.oauth2.resourceserver.jwt.public-key-
location property, where the value needs to point to a file containing the public key
in the PEM-encoded x509 format.
The same properties are applicable for both servlet and reactive applications.
Alternatively, you can define your own JwtDecoder bean for servlet applications or a
ReactiveJwtDecoder for reactive applications.
In cases where opaque tokens are used instead of JWTs, you can configure the following properties
to validate tokens through introspection:
Properties
spring.security.oauth2.resourceserver.opaquetoken.introspection-
uri=https://example.com/check-token
spring.security.oauth2.resourceserver.opaquetoken.client-id=my-client-id
spring.security.oauth2.resourceserver.opaquetoken.client-secret=my-client-secret
Yaml
spring:
security:
oauth2:
resourceserver:
opaquetoken:
introspection-uri: "https://example.com/check-token"
client-id: "my-client-id"
client-secret: "my-client-secret"
Again, the same properties are applicable for both servlet and reactive applications.
Alternatively, you can define your own OpaqueTokenIntrospector bean for servlet applications or a
ReactiveOpaqueTokenIntrospector for reactive applications.
Authorization Server
310
Properties
spring.security.oauth2.authorizationserver.client.my-client-1.registration.client-
id=abcd
spring.security.oauth2.authorizationserver.client.my-client-1.registration.client-
secret={noop}secret1
spring.security.oauth2.authorizationserver.client.my-client-1.registration.client-
authentication-methods[0]=client_secret_basic
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.authorization-grant-types[0]=authorization_code
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.authorization-grant-types[1]=refresh_token
spring.security.oauth2.authorizationserver.client.my-client-1.registration.redirect-
uris[0]=https://my-client-1.com/login/oauth2/code/abcd
spring.security.oauth2.authorizationserver.client.my-client-1.registration.redirect-
uris[1]=https://my-client-1.com/authorized
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.scopes[0]=openid
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.scopes[1]=profile
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.scopes[2]=email
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.scopes[3]=phone
spring.security.oauth2.authorizationserver.client.my-client-
1.registration.scopes[4]=address
spring.security.oauth2.authorizationserver.client.my-client-1.require-authorization-
consent=true
spring.security.oauth2.authorizationserver.client.my-client-2.registration.client-
id=efgh
spring.security.oauth2.authorizationserver.client.my-client-2.registration.client-
secret={noop}secret2
spring.security.oauth2.authorizationserver.client.my-client-2.registration.client-
authentication-methods[0]=client_secret_jwt
spring.security.oauth2.authorizationserver.client.my-client-
2.registration.authorization-grant-types[0]=client_credentials
spring.security.oauth2.authorizationserver.client.my-client-
2.registration.scopes[0]=user.read
spring.security.oauth2.authorizationserver.client.my-client-
2.registration.scopes[1]=user.write
spring.security.oauth2.authorizationserver.client.my-client-2.jwk-set-uri=https://my-
client-2.com/jwks
spring.security.oauth2.authorizationserver.client.my-client-2.token-endpoint-
authentication-signing-algorithm=RS256
311
Yaml
spring:
security:
oauth2:
authorizationserver:
client:
my-client-1:
registration:
client-id: "abcd"
client-secret: "{noop}secret1"
client-authentication-methods:
- "client_secret_basic"
authorization-grant-types:
- "authorization_code"
- "refresh_token"
redirect-uris:
- "https://my-client-1.com/login/oauth2/code/abcd"
- "https://my-client-1.com/authorized"
scopes:
- "openid"
- "profile"
- "email"
- "phone"
- "address"
require-authorization-consent: true
my-client-2:
registration:
client-id: "efgh"
client-secret: "{noop}secret2"
client-authentication-methods:
- "client_secret_jwt"
authorization-grant-types:
- "client_credentials"
scopes:
- "user.read"
- "user.write"
jwk-set-uri: "https://my-client-2.com/jwks"
token-endpoint-authentication-signing-algorithm: "RS256"
The auto-configuration Spring Boot provides for Spring Authorization Server is designed for getting
started quickly. Most applications will require customization and will want to define several beans
to override auto-configuration.
The following components can be defined as beans to override auto-configuration specific to Spring
Authorization Server:
312
• RegisteredClientRepository
• AuthorizationServerSettings
• SecurityFilterChain
• com.nimbusds.jose.jwk.source.JWKSource<com.nimbusds.jose.proc.SecurityContext>
• JwtDecoder
Additional information can be found in the Getting Started chapter of the Spring Authorization
Server Reference Guide.
Relying Party
A relying party registration represents a paired configuration between an Identity Provider, IDP,
and a Service Provider, SP. You can register multiple relying parties under the
spring.security.saml2.relyingparty prefix, as shown in the following example:
313
Properties
spring.security.saml2.relyingparty.registration.my-relying-
party1.signing.credentials[0].private-key-location=path-to-private-key
spring.security.saml2.relyingparty.registration.my-relying-
party1.signing.credentials[0].certificate-location=path-to-certificate
spring.security.saml2.relyingparty.registration.my-relying-
party1.decryption.credentials[0].private-key-location=path-to-private-key
spring.security.saml2.relyingparty.registration.my-relying-
party1.decryption.credentials[0].certificate-location=path-to-certificate
spring.security.saml2.relyingparty.registration.my-relying-
party1.singlelogout.url=https://myapp/logout/saml2/slo
spring.security.saml2.relyingparty.registration.my-relying-
party1.singlelogout.response-url=https://remoteidp2.slo.url
spring.security.saml2.relyingparty.registration.my-relying-
party1.singlelogout.binding=POST
spring.security.saml2.relyingparty.registration.my-relying-
party1.assertingparty.verification.credentials[0].certificate-location=path-to-
verification-cert
spring.security.saml2.relyingparty.registration.my-relying-
party1.assertingparty.entity-id=remote-idp-entity-id1
spring.security.saml2.relyingparty.registration.my-relying-party1.assertingparty.sso-
url=https://remoteidp1.sso.url
spring.security.saml2.relyingparty.registration.my-relying-
party2.signing.credentials[0].private-key-location=path-to-private-key
spring.security.saml2.relyingparty.registration.my-relying-
party2.signing.credentials[0].certificate-location=path-to-certificate
spring.security.saml2.relyingparty.registration.my-relying-
party2.decryption.credentials[0].private-key-location=path-to-private-key
spring.security.saml2.relyingparty.registration.my-relying-
party2.decryption.credentials[0].certificate-location=path-to-certificate
spring.security.saml2.relyingparty.registration.my-relying-
party2.assertingparty.verification.credentials[0].certificate-location=path-to-other-
verification-cert
spring.security.saml2.relyingparty.registration.my-relying-
party2.assertingparty.entity-id=remote-idp-entity-id2
spring.security.saml2.relyingparty.registration.my-relying-party2.assertingparty.sso-
url=https://remoteidp2.sso.url
spring.security.saml2.relyingparty.registration.my-relying-
party2.assertingparty.singlelogout.url=https://remoteidp2.slo.url
spring.security.saml2.relyingparty.registration.my-relying-
party2.assertingparty.singlelogout.response-url=https://myapp/logout/saml2/slo
spring.security.saml2.relyingparty.registration.my-relying-
party2.assertingparty.singlelogout.binding=POST
314
Yaml
spring:
security:
saml2:
relyingparty:
registration:
my-relying-party1:
signing:
credentials:
- private-key-location: "path-to-private-key"
certificate-location: "path-to-certificate"
decryption:
credentials:
- private-key-location: "path-to-private-key"
certificate-location: "path-to-certificate"
singlelogout:
url: "https://myapp/logout/saml2/slo"
response-url: "https://remoteidp2.slo.url"
binding: "POST"
assertingparty:
verification:
credentials:
- certificate-location: "path-to-verification-cert"
entity-id: "remote-idp-entity-id1"
sso-url: "https://remoteidp1.sso.url"
my-relying-party2:
signing:
credentials:
- private-key-location: "path-to-private-key"
certificate-location: "path-to-certificate"
decryption:
credentials:
- private-key-location: "path-to-private-key"
certificate-location: "path-to-certificate"
assertingparty:
verification:
credentials:
- certificate-location: "path-to-other-verification-cert"
entity-id: "remote-idp-entity-id2"
sso-url: "https://remoteidp2.sso.url"
singlelogout:
url: "https://remoteidp2.slo.url"
response-url: "https://myapp/logout/saml2/slo"
binding: "POST"
315
logout responses to, to use a different pattern, you need to provide configuration to process that
custom pattern. For example, for servlet applications, you can add your own SecurityFilterChain
that resembles the following:
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MySamlRelyingPartyConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.authorizeHttpRequests((requests) ->
requests.anyRequest().authenticated());
http.saml2Login(withDefaults());
http.saml2Logout((saml2) -> saml2.logoutRequest((request) ->
request.logoutUrl("/SLOService.saml2"))
.logoutResponse((response) -> response.logoutUrl("/SLOService.saml2")));
return http.build();
}
• Redis
• JDBC
• Hazelcast
• MongoDB
Additionally, Spring Boot for Apache Geode provides auto-configuration for using Apache Geode as
a session store.
If a single Spring Session module is present on the classpath, Spring Boot uses that store
implementation automatically. If you have more than one implementation, Spring Boot uses the
following order for choosing a specific implementation:
1. Redis
316
2. JDBC
3. Hazelcast
4. MongoDB
5. If none of Redis, JDBC, Hazelcast and MongoDB are available, we do not configure a
SessionRepository.
When building a reactive web application, the following stores can be auto-configured:
• Redis
• MongoDB
Similar to the servlet configuration, if you have more than one implementation, Spring Boot uses
the following order for choosing a specific implementation:
1. Redis
2. MongoDB
Each store has specific additional settings. For instance, it is possible to customize the name of the
table for the JDBC store, as shown in the following example:
Properties
spring.session.jdbc.table-name=SESSIONS
Yaml
spring:
session:
jdbc:
table-name: "SESSIONS"
For setting the timeout of the session you can use the spring.session.timeout property. If that
property is not set with a servlet web application, the auto-configuration falls back to the value of
server.servlet.session.timeout.
You can take control over Spring Session’s configuration using @Enable*HttpSession (servlet) or
@Enable*WebSession (reactive). This will cause the auto-configuration to back off. Spring Session can
then be configured using the annotation’s attributes rather than the previously described
configuration properties.
317
You’ll need the spring-boot-starter-graphql starter at a minimum. Because GraphQL is transport-
agnostic, you’ll also need to have one or more additional starters in your application to expose your
GraphQL API over the web:
A Spring GraphQL application requires a defined schema at startup. By default, you can write
".graphqls" or ".gqls" schema files under src/main/resources/graphql/** and Spring Boot will pick
them up automatically. You can customize the locations with spring.graphql.schema.locations and
the file extensions with spring.graphql.schema.file-extensions.
If you want Spring Boot to detect schema files in all your application modules and
NOTE dependencies for that location, you can set spring.graphql.schema.locations to
"classpath*:graphql/**/" (note the classpath*: prefix).
In the following sections, we’ll consider this sample GraphQL schema, defining two types and two
queries:
318
type Query {
greeting(name: String! = "Spring"): String!
project(slug: ID!): Project
}
enum ProjectStatus {
""" Actively supported by the Spring team """
ACTIVE
""" Supported by the community """
COMMUNITY
""" Prototype, not officially supported yet """
INCUBATING
""" Project being retired, in maintenance mode """
ATTIC
""" End-Of-Lifed """
EOL
}
The GraphQL Java RuntimeWiring.Builder can be used to register custom scalar types, directives,
type resolvers, DataFetcher, and more. You can declare RuntimeWiringConfigurer beans in your
Spring config to get access to the RuntimeWiring.Builder. Spring Boot detects such beans and adds
them to the GraphQlSource builder.
Typically, however, applications will not implement DataFetcher directly and will instead create
annotated controllers. Spring Boot will automatically detect @Controller classes with annotated
handler methods and register those as DataFetchers. Here’s a sample implementation for our
greeting query with a @Controller class:
319
Java
import org.springframework.graphql.data.method.annotation.Argument;
import org.springframework.graphql.data.method.annotation.QueryMapping;
import org.springframework.stereotype.Controller;
@Controller
public class GreetingController {
@QueryMapping
public String greeting(@Argument String name) {
return "Hello, " + name + "!";
}
Kotlin
import org.springframework.graphql.data.method.annotation.Argument
import org.springframework.graphql.data.method.annotation.QueryMapping
import org.springframework.stereotype.Controller
@Controller
class GreetingController {
@QueryMapping
fun greeting(@Argument name: String): String {
return "Hello, $name!"
}
Spring Data offers support for both Querydsl and QueryByExample repositories. Spring GraphQL
can configure Querydsl and QueryByExample repositories as DataFetcher.
Spring Data repositories annotated with @GraphQlRepository and extending one of:
• QuerydslPredicateExecutor
• ReactiveQuerydslPredicateExecutor
• QueryByExampleExecutor
• ReactiveQueryByExampleExecutor
are detected by Spring Boot and considered as candidates for DataFetcher for matching top-level
queries.
320
8.6.4. Transports
The GraphQL HTTP endpoint is at HTTP POST /graphql by default. The path can be customized with
spring.graphql.path.
The HTTP endpoint for both Spring MVC and Spring WebFlux is provided by a
RouterFunction bean with an @Order of 0. If you define your own RouterFunction beans,
TIP
you may want to add appropriate @Order annotations to ensure that they are sorted
correctly.
Spring GraphQL provides a Web Interception model. This is quite useful for retrieving information
from an HTTP request header and set it in the GraphQL context or fetching information from the
same context and writing it to a response header. With Spring Boot, you can declare a
WebInterceptor bean to have it registered with the web transport.
Spring MVC and Spring WebFlux support CORS (Cross-Origin Resource Sharing) requests. CORS is a
critical part of the web config for GraphQL applications that are accessed from browsers using
different domains.
Spring Boot supports many configuration properties under the spring.graphql.cors.* namespace;
here’s a short configuration sample:
Properties
spring.graphql.cors.allowed-origins=https://example.org
spring.graphql.cors.allowed-methods=GET,POST
spring.graphql.cors.max-age=1800s
Yaml
spring:
graphql:
cors:
allowed-origins: "https://example.org"
allowed-methods: GET,POST
max-age: 1800s
RSocket
RSocket is also supported as a transport, on top of WebSocket or TCP. Once the RSocket server is
321
configured, we can configure our GraphQL handler on a particular route using
spring.graphql.rsocket.mapping. For example, configuring that mapping as "graphql" means we can
use that as a route when sending requests with the RSocketGraphQlClient.
Spring Boot auto-configures a RSocketGraphQlClient.Builder<?> bean that you can inject in your
components:
Java
@Component
public class RSocketGraphQlClientExample {
Kotlin
@Component
class RSocketGraphQlClientExample(private val builder:
RSocketGraphQlClient.Builder<*>) {
Java
Kotlin
322
8.6.5. Exception Handling
Spring GraphQL offers infrastructure for helping developers when consuming or developing a
GraphQL API.
Spring GraphQL ships with a default GraphiQL page that is exposed at "/graphiql" by default. This
page is disabled by default and can be turned on with the spring.graphql.graphiql.enabled
property. Many applications exposing such a page will prefer a custom build. A default
implementation is very useful during development, this is why it is exposed automatically with
spring-boot-devtools during development.
You can also choose to expose the GraphQL schema in text format at /graphql/schema when the
spring.graphql.schema.printer.enabled property is enabled.
You can take control of Spring HATEOAS’s configuration by using @EnableHypermediaSupport. Note
that doing so disables the ObjectMapper customization described earlier.
323
Chapter 9. Data
Spring Boot integrates with a number of data technologies, both SQL and NoSQL.
See the “How-to” section for more advanced examples, typically to take full control
TIP
over the configuration of the DataSource.
Spring Boot can auto-configure embedded H2, HSQL, and Derby databases. You need not provide
any connection URLs. You need only include a build dependency to the embedded database that
you want to use. If there are multiple embedded databases on the classpath, set the
spring.datasource.embedded-database-connection configuration property to control which one is
used. Setting the property to none disables auto-configuration of an embedded database.
If you are using this feature in your tests, you may notice that the same database is
reused by your whole test suite regardless of the number of application contexts
NOTE
that you use. If you want to make sure that each context has a separate embedded
database, you should set spring.datasource.generate-unique-name to true.
324
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.hsqldb</groupId>
<artifactId>hsqldb</artifactId>
<scope>runtime</scope>
</dependency>
If, for whatever reason, you do configure the connection URL for an embedded
database, take care to ensure that the database’s automatic shutdown is disabled. If
you use H2, you should use DB_CLOSE_ON_EXIT=FALSE to do so. If you use HSQLDB, you
TIP
should ensure that shutdown=true is not used. Disabling the database’s automatic
shutdown lets Spring Boot control when the database is closed, thereby ensuring that
it happens once access to the database is no longer needed.
DataSource Configuration
Properties
spring.datasource.url=jdbc:mysql://localhost/test
spring.datasource.username=dbuser
spring.datasource.password=dbpass
Yaml
spring:
datasource:
url: "jdbc:mysql://localhost/test"
username: "dbuser"
password: "dbpass"
You should at least specify the URL by setting the spring.datasource.url property.
NOTE
Otherwise, Spring Boot tries to auto-configure an embedded database.
325
Spring Boot can deduce the JDBC driver class for most databases from the URL. If you
TIP need to specify a specific class, you can use the spring.datasource.driver-class-name
property.
See DataSourceProperties for more of the supported options. These are the standard options that
work regardless of the actual implementation. It is also possible to fine-tune implementation-
specific settings by using their respective prefix (spring.datasource.hikari.*,
spring.datasource.tomcat.*, spring.datasource.dbcp2.*, and spring.datasource.oracleucp.*). See the
documentation of the connection pool implementation you are using for more details.
For instance, if you use the Tomcat connection pool, you could customize many additional settings,
as shown in the following example:
Properties
spring.datasource.tomcat.max-wait=10000
spring.datasource.tomcat.max-active=50
spring.datasource.tomcat.test-on-borrow=true
Yaml
spring:
datasource:
tomcat:
max-wait: 10000
max-active: 50
test-on-borrow: true
This will set the pool to wait 10000ms before throwing an exception if no connection is available,
limit the maximum number of connections to 50 and validate the connection before borrowing it
from the pool.
Spring Boot uses the following algorithm for choosing a specific implementation:
1. We prefer HikariCP for its performance and concurrency. If HikariCP is available, we always
choose it.
4. If none of HikariCP, Tomcat, and DBCP2 are available and if Oracle UCP is available, we use it.
326
If you use the spring-boot-starter-jdbc or spring-boot-starter-data-jpa “starters”,
NOTE
you automatically get a dependency to HikariCP.
You can bypass that algorithm completely and specify the connection pool to use by setting the
spring.datasource.type property. This is especially important if you run your application in a
Tomcat container, as tomcat-jdbc is provided by default.
Additional connection pools can always be configured manually, using DataSourceBuilder. If you
define your own DataSource bean, auto-configuration does not occur. The following connection
pools are supported by DataSourceBuilder:
• HikariCP
• Commons DBCP2
• H2 JdbcDataSource
• PostgreSQL PGSimpleDataSource
• C3P0
If you deploy your Spring Boot application to an Application Server, you might want to configure
and manage your DataSource by using your Application Server’s built-in features and access it by
using JNDI.
Properties
spring.datasource.jndi-name=java:jboss/datasources/customers
Yaml
spring:
datasource:
jndi-name: "java:jboss/datasources/customers"
Spring’s JdbcTemplate and NamedParameterJdbcTemplate classes are auto-configured, and you can
@Autowire them directly into your own beans, as shown in the following example:
327
Java
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.jdbc.core.JdbcTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val jdbcTemplate: JdbcTemplate) {
fun doSomething() {
jdbcTemplate.execute("delete from customer")
}
You can customize some properties of the template by using the spring.jdbc.template.* properties,
as shown in the following example:
Properties
spring.jdbc.template.max-rows=500
Yaml
spring:
jdbc:
template:
max-rows: 500
328
The NamedParameterJdbcTemplate reuses the same JdbcTemplate instance behind the
NOTE scenes. If more than one JdbcTemplate is defined and no primary candidate exists,
the NamedParameterJdbcTemplate is not auto-configured.
The Java Persistence API is a standard technology that lets you “map” objects to relational
databases. The spring-boot-starter-data-jpa POM provides a quick way to get started. It provides
the following key dependencies:
We do not go into too many details of JPA or Spring Data here. You can follow the
TIP “Accessing Data with JPA” guide from spring.io and read the Spring Data JPA and
Hibernate reference documentation.
Entity Classes
Traditionally, JPA “Entity” classes are specified in a persistence.xml file. With Spring Boot, this file is
not necessary and “Entity Scanning” is used instead. By default, all packages below your main
configuration class (the one annotated with @EnableAutoConfiguration or @SpringBootApplication)
are searched.
Any classes annotated with @Entity, @Embeddable, or @MappedSuperclass are considered. A typical
entity class resembles the following example:
329
Java
import java.io.Serializable;
import jakarta.persistence.Column;
import jakarta.persistence.Entity;
import jakarta.persistence.GeneratedValue;
import jakarta.persistence.Id;
@Entity
public class City implements Serializable {
@Id
@GeneratedValue
private Long id;
@Column(nullable = false)
private String name;
@Column(nullable = false)
private String state;
protected City() {
// no-args constructor required by JPA spec
// this one is protected since it should not be used directly
}
// ... etc
330
Kotlin
import jakarta.persistence.Column
import jakarta.persistence.Entity
import jakarta.persistence.GeneratedValue
import jakarta.persistence.Id
import java.io.Serializable
@Entity
class City : Serializable {
@Id
@GeneratedValue
private val id: Long? = null
@Column(nullable = false)
var name: String? = null
private set
// ... etc
@Column(nullable = false)
var state: String? = null
private set
protected constructor() {
// no-args constructor required by JPA spec
// this one is protected since it should not be used directly
}
You can customize entity scanning locations by using the @EntityScan annotation. See
TIP
the “Separate @Entity Definitions from Spring Configuration” how-to.
Spring Data JPA repositories are interfaces that you can define to access data. JPA queries are
created automatically from your method names. For example, a CityRepository interface might
declare a findAllByState(String state) method to find all the cities in a given state.
For more complex queries, you can annotate your method with Spring Data’s Query annotation.
Spring Data repositories usually extend from the Repository or CrudRepository interfaces. If you use
331
auto-configuration, repositories are searched from the package containing your main configuration
class (the one annotated with @EnableAutoConfiguration or @SpringBootApplication) down.
The following example shows a typical Spring Data repository interface definition:
Java
import org.springframework.boot.docs.data.sql.jpaandspringdata.entityclasses.City;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.repository.Repository;
Kotlin
import org.springframework.boot.docs.data.sql.jpaandspringdata.entityclasses.City
import org.springframework.data.domain.Page
import org.springframework.data.domain.Pageable
import org.springframework.data.repository.Repository
Spring Data JPA repositories support three different modes of bootstrapping: default, deferred, and
lazy. To enable deferred or lazy bootstrapping, set the spring.data.jpa.repositories.bootstrap-mode
property to deferred or lazy respectively. When using deferred or lazy bootstrapping, the auto-
configured EntityManagerFactoryBuilder will use the context’s AsyncTaskExecutor, if any, as the
bootstrap executor. If more than one exists, the one named applicationTaskExecutor will be used.
When using deferred or lazy bootstrapping, make sure to defer any access to the JPA
infrastructure after the application context bootstrap phase. You can use
NOTE SmartInitializingSingleton to invoke any initialization that requires the JPA
infrastructure. For JPA components (such as converters) that are created as Spring
beans, use ObjectProvider to delay the resolution of dependencies, if any.
332
We have barely scratched the surface of Spring Data JPA. For complete details, see the
TIP
Spring Data JPA reference documentation.
If Spring Data Envers is available, JPA repositories are auto-configured to support typical Envers
queries.
To use Spring Data Envers, make sure your repository extends from RevisionRepository as shown in
the following example:
Java
import org.springframework.boot.docs.data.sql.jpaandspringdata.entityclasses.Country;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.repository.Repository;
import org.springframework.data.repository.history.RevisionRepository;
Kotlin
import org.springframework.boot.docs.data.sql.jpaandspringdata.entityclasses.Country
import org.springframework.data.domain.Page
import org.springframework.data.domain.Pageable
import org.springframework.data.repository.Repository
import org.springframework.data.repository.history.RevisionRepository
interface CountryRepository :
RevisionRepository<Country?, Long?, Int>,
Repository<Country?, Long?> {
NOTE For more details, check the Spring Data Envers reference documentation.
By default, JPA databases are automatically created only if you use an embedded database (H2,
HSQL, or Derby). You can explicitly configure JPA settings by using spring.jpa.* properties. For
example, to create and drop tables you can add the following line to your application.properties:
333
Properties
spring.jpa.hibernate.ddl-auto=create-drop
Yaml
spring:
jpa:
hibernate.ddl-auto: "create-drop"
Hibernate’s own internal property name for this (if you happen to remember it
better) is hibernate.hbm2ddl.auto. You can set it, along with other Hibernate native
NOTE properties, by using spring.jpa.properties.* (the prefix is stripped before adding
them to the entity manager). The following line shows an example of setting JPA
properties for Hibernate:
Properties
spring.jpa.properties.hibernate[globally_quoted_identifiers]=true
Yaml
spring:
jpa:
properties:
hibernate:
"globally_quoted_identifiers": "true"
The line in the preceding example passes a value of true for the
hibernate.globally_quoted_identifiers property to the Hibernate entity manager.
By default, the DDL execution (or validation) is deferred until the ApplicationContext has started.
There is also a spring.jpa.generate-ddl flag, but it is not used if Hibernate auto-configuration is
active, because the ddl-auto settings are more fine-grained.
Spring Data includes repository support for JDBC and will automatically generate SQL for the
methods on CrudRepository. For more advanced queries, a @Query annotation is provided.
334
Spring Boot will auto-configure Spring Data’s JDBC repositories when the necessary dependencies
are on the classpath. They can be added to your project with a single dependency on spring-boot-
starter-data-jdbc. If necessary, you can take control of Spring Data JDBC’s configuration by adding
the @EnableJdbcRepositories annotation or an AbstractJdbcConfiguration subclass to your
application.
TIP For complete details of Spring Data JDBC, see the reference documentation.
The H2 database provides a browser-based console that Spring Boot can auto-configure for you. The
console is auto-configured when the following conditions are met:
If you are not using Spring Boot’s developer tools but would still like to make use of
TIP H2’s console, you can configure the spring.h2.console.enabled property with a value
of true.
The H2 console is only intended for use during development, so you should take
NOTE
care to ensure that spring.h2.console.enabled is not set to true in production.
By default, the console is available at /h2-console. You can customize the console’s path by using the
spring.h2.console.path property.
H2 Console uses frames and, as it is intended for development only, does not implement CSRF
protection measures. If your application uses Spring Security, you need to configure it to
More information on CSRF and the header X-Frame-Options can be found in the Spring Security
Reference Guide.
335
Java
import org.springframework.boot.autoconfigure.security.servlet.PathRequest;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;
import org.springframework.core.Ordered;
import org.springframework.core.annotation.Order;
import org.springframework.security.config.Customizer;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Profile("dev")
@Configuration(proxyBeanMethods = false)
public class DevProfileSecurityConfiguration {
@Bean
@Order(Ordered.HIGHEST_PRECEDENCE)
SecurityFilterChain h2ConsoleSecurityFilterChain(HttpSecurity http) throws
Exception {
http.securityMatcher(PathRequest.toH2Console());
http.authorizeHttpRequests(yourCustomAuthorization());
http.csrf((csrf) -> csrf.disable());
http.headers((headers) -> headers.frameOptions((frame) ->
frame.sameOrigin()));
return http.build();
}
336
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Profile
import org.springframework.core.Ordered
import org.springframework.core.annotation.Order
import org.springframework.security.config.Customizer
import org.springframework.security.config.annotation.web.builders.HttpSecurity
import org.springframework.security.web.SecurityFilterChain
@Profile("dev")
@Configuration(proxyBeanMethods = false)
class DevProfileSecurityConfiguration {
@Bean
@Order(Ordered.HIGHEST_PRECEDENCE)
fun h2ConsoleSecurityFilterChain(http: HttpSecurity): SecurityFilterChain {
return http.authorizeHttpRequests(yourCustomAuthorization())
.csrf { csrf -> csrf.disable() }
.headers { headers -> headers.frameOptions { frameOptions ->
frameOptions.sameOrigin() } }
.build()
}
jOOQ Object Oriented Querying (jOOQ) is a popular product from Data Geekery which generates
Java code from your database and lets you build type-safe SQL queries through its fluent API. Both
the commercial and open source editions can be used with Spring Boot.
Code Generation
In order to use jOOQ type-safe queries, you need to generate Java classes from your database
schema. You can follow the instructions in the jOOQ user manual. If you use the jooq-codegen-maven
plugin and you also use the spring-boot-starter-parent “parent POM”, you can safely omit the
plugin’s <version> tag. You can also use Spring Boot-defined version variables (such as h2.version)
to declare the plugin’s database dependency. The following listing shows an example:
337
<plugin>
<groupId>org.jooq</groupId>
<artifactId>jooq-codegen-maven</artifactId>
<executions>
...
</executions>
<dependencies>
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
<version>${h2.version}</version>
</dependency>
</dependencies>
<configuration>
<jdbc>
<driver>org.h2.Driver</driver>
<url>jdbc:h2:~/yourdatabase</url>
</jdbc>
<generator>
...
</generator>
</configuration>
</plugin>
Using DSLContext
The fluent API offered by jOOQ is initiated through the org.jooq.DSLContext interface. Spring Boot
auto-configures a DSLContext as a Spring Bean and connects it to your application DataSource. To use
the DSLContext, you can inject it, as shown in the following example:
338
Java
import java.util.GregorianCalendar;
import java.util.List;
import org.jooq.DSLContext;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.jooq.DSLContext
import org.springframework.stereotype.Component
import java.util.GregorianCalendar
@Component
class MyBean(private val create: DSLContext) {
TIP The jOOQ manual tends to use a variable named create to hold the DSLContext.
You can then use the DSLContext to construct your queries, as shown in the following example:
Java
339
Kotlin
Unless the spring.jooq.sql-dialect property has been configured, Spring Boot determines the SQL
dialect to use for your datasource. If Spring Boot could not detect the dialect, it uses DEFAULT.
Spring Boot can only auto-configure dialects supported by the open source version
NOTE
of jOOQ.
Customizing jOOQ
You can also create your own org.jooq.Configuration @Bean if you want to take complete control of
the jOOQ configuration.
The Reactive Relational Database Connectivity (R2DBC) project brings reactive programming APIs
to relational databases. R2DBC’s io.r2dbc.spi.Connection provides a standard method of working
with non-blocking database connections. Connections are provided by using a ConnectionFactory,
similar to a DataSource with jdbc.
Properties
spring.r2dbc.url=r2dbc:postgresql://localhost/test
spring.r2dbc.username=dbuser
spring.r2dbc.password=dbpass
340
Yaml
spring:
r2dbc:
url: "r2dbc:postgresql://localhost/test"
username: "dbuser"
password: "dbpass"
You do not need to specify a driver class name, since Spring Boot obtains the driver
TIP
from R2DBC’s Connection Factory discovery.
At least the url should be provided. Information specified in the URL takes
NOTE precedence over individual properties, that is name, username, password and pooling
options.
To customize the connections created by a ConnectionFactory, that is, set specific parameters that
you do not want (or cannot) configure in your central database configuration, you can use a
ConnectionFactoryOptionsBuilderCustomizer @Bean. The following example shows how to manually
override the database port while the rest of the options is taken from the application configuration:
Java
import io.r2dbc.spi.ConnectionFactoryOptions;
import
org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryOptionsBuilderCustomizer
;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyR2dbcConfiguration {
@Bean
public ConnectionFactoryOptionsBuilderCustomizer connectionFactoryPortCustomizer()
{
return (builder) -> builder.option(ConnectionFactoryOptions.PORT, 5432);
}
341
Kotlin
import io.r2dbc.spi.ConnectionFactoryOptions
import
org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryOptionsBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyR2dbcConfiguration {
@Bean
fun connectionFactoryPortCustomizer(): ConnectionFactoryOptionsBuilderCustomizer {
return ConnectionFactoryOptionsBuilderCustomizer { builder ->
builder.option(ConnectionFactoryOptions.PORT, 5432)
}
}
The following examples show how to set some PostgreSQL connection options:
Java
import java.util.HashMap;
import java.util.Map;
import io.r2dbc.postgresql.PostgresqlConnectionFactoryProvider;
import
org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryOptionsBuilderCustomizer
;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyPostgresR2dbcConfiguration {
@Bean
public ConnectionFactoryOptionsBuilderCustomizer postgresCustomizer() {
Map<String, String> options = new HashMap<>();
options.put("lock_timeout", "30s");
options.put("statement_timeout", "60s");
return (builder) ->
builder.option(PostgresqlConnectionFactoryProvider.OPTIONS, options);
}
342
Kotlin
import io.r2dbc.postgresql.PostgresqlConnectionFactoryProvider
import
org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryOptionsBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyPostgresR2dbcConfiguration {
@Bean
fun postgresCustomizer(): ConnectionFactoryOptionsBuilderCustomizer {
val options: MutableMap<String, String> = HashMap()
options["lock_timeout"] = "30s"
options["statement_timeout"] = "60s"
return ConnectionFactoryOptionsBuilderCustomizer { builder ->
builder.option(PostgresqlConnectionFactoryProvider.OPTIONS, options)
}
}
When a ConnectionFactory bean is available, the regular JDBC DataSource auto-configuration backs
off. If you want to retain the JDBC DataSource auto-configuration, and are comfortable with the risk
of using the blocking JDBC API in a reactive application, add
@Import(DataSourceAutoConfiguration.class) on a @Configuration class in your application to re-
enable it.
Similarly to the JDBC support, Spring Boot can automatically configure an embedded database for
reactive usage. You need not provide any connection URLs. You need only include a build
dependency to the embedded database that you want to use, as shown in the following example:
<dependency>
<groupId>io.r2dbc</groupId>
<artifactId>r2dbc-h2</artifactId>
<scope>runtime</scope>
</dependency>
If you are using this feature in your tests, you may notice that the same database is
reused by your whole test suite regardless of the number of application contexts
NOTE
that you use. If you want to make sure that each context has a separate embedded
database, you should set spring.r2dbc.generate-unique-name to true.
343
Using DatabaseClient
A DatabaseClient bean is auto-configured, and you can @Autowire it directly into your own beans, as
shown in the following example:
Java
import java.util.Map;
import reactor.core.publisher.Flux;
import org.springframework.r2dbc.core.DatabaseClient;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.r2dbc.core.DatabaseClient
import org.springframework.stereotype.Component
import reactor.core.publisher.Flux
@Component
class MyBean(private val databaseClient: DatabaseClient) {
Spring Data R2DBC repositories are interfaces that you can define to access data. Queries are
created automatically from your method names. For example, a CityRepository interface might
declare a findAllByState(String state) method to find all the cities in a given state.
344
For more complex queries, you can annotate your method with Spring Data’s Query annotation.
Spring Data repositories usually extend from the Repository or CrudRepository interfaces. If you use
auto-configuration, repositories are searched from the package containing your main configuration
class (the one annotated with @EnableAutoConfiguration or @SpringBootApplication) down.
The following example shows a typical Spring Data repository interface definition:
Java
import reactor.core.publisher.Mono;
import org.springframework.data.repository.Repository;
Kotlin
import org.springframework.data.repository.Repository
import reactor.core.publisher.Mono
We have barely scratched the surface of Spring Data R2DBC. For complete details, see
TIP
the Spring Data R2DBC reference documentation.
• MongoDB
• Neo4J
• Elasticsearch
• Redis
• GemFire or Geode
• Cassandra
• Couchbase
345
• LDAP
Spring Boot provides auto-configuration for Redis, MongoDB, Neo4j, Elasticsearch, Cassandra,
Couchbase, LDAP and InfluxDB. Additionally, Spring Boot for Apache Geode provides auto-
configuration for Apache Geode. You can make use of the other projects, but you must configure
them yourself. See the appropriate reference documentation at spring.io/projects/spring-data.
9.2.1. Redis
Redis is a cache, message broker, and richly-featured key-value store. Spring Boot offers basic auto-
configuration for the Lettuce and Jedis client libraries and the abstractions on top of them provided
by Spring Data Redis.
Connecting to Redis
Java
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
346
Kotlin
import org.springframework.data.redis.core.StringRedisTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val template: StringRedisTemplate) {
By default, the instance tries to connect to a Redis server at localhost:6379. You can specify custom
connection details using spring.data.redis.* properties, as shown in the following example:
Properties
spring.data.redis.host=localhost
spring.data.redis.port=6379
spring.data.redis.database=0
spring.data.redis.username=user
spring.data.redis.password=secret
Yaml
spring:
data:
redis:
host: "localhost"
port: 6379
database: 0
username: "user"
password: "secret"
If you add your own @Bean of any of the auto-configured types, it replaces the default (except in the
case of RedisTemplate, when the exclusion is based on the bean name, redisTemplate, not its type).
347
The auto-configured RedisConnectionFactory can be configured to use SSL for communication with
the server by setting the properties as shown in this example:
Properties
spring.data.redis.ssl.enabled=true
Yaml
spring:
data:
redis:
ssl:
enabled: true
Custom SSL trust material can be configured in an SSL bundle and applied to the
RedisConnectionFactory as shown in this example:
Properties
spring.data.redis.ssl.bundle=example
Yaml
spring:
data:
redis:
ssl:
bundle: "example"
9.2.2. MongoDB
MongoDB is an open-source NoSQL document database that uses a JSON-like schema instead of
traditional table-based relational data. Spring Boot offers several conveniences for working with
MongoDB, including the spring-boot-starter-data-mongodb and spring-boot-starter-data-mongodb-
reactive “Starters”.
348
Java
import com.mongodb.client.MongoCollection;
import com.mongodb.client.MongoDatabase;
import org.bson.Document;
import org.springframework.data.mongodb.MongoDatabaseFactory;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import com.mongodb.client.MongoCollection
import org.bson.Document
import org.springframework.data.mongodb.MongoDatabaseFactory
import org.springframework.stereotype.Component
@Component
class MyBean(private val mongo: MongoDatabaseFactory) {
If you have defined your own MongoClient, it will be used to auto-configure a suitable
MongoDatabaseFactory.
The auto-configured MongoClient is created using a MongoClientSettings bean. If you have defined
your own MongoClientSettings, it will be used without modification and the spring.data.mongodb
properties will be ignored. Otherwise a MongoClientSettings will be auto-configured and will have
the spring.data.mongodb properties applied to it. In either case, you can declare one or more
349
MongoClientSettingsBuilderCustomizer beans to fine-tune the MongoClientSettings configuration.
Each will be called in order with the MongoClientSettings.Builder that is used to build the
MongoClientSettings.
You can set the spring.data.mongodb.uri property to change the URL and configure additional
settings such as the replica set, as shown in the following example:
Properties
spring.data.mongodb.uri=mongodb://user:secret@mongoserver1.example.com:27017,mongoserv
er2.example.com:23456/test
Yaml
spring:
data:
mongodb:
uri:
"mongodb://user:secret@mongoserver1.example.com:27017,mongoserver2.example.com:23456/t
est"
Alternatively, you can specify connection details using discrete properties. For example, you might
declare the following settings in your application.properties:
Properties
spring.data.mongodb.host=mongoserver1.example.com
spring.data.mongodb.port=27017
spring.data.mongodb.additional-hosts[0]=mongoserver2.example.com:23456
spring.data.mongodb.database=test
spring.data.mongodb.username=user
spring.data.mongodb.password=secret
Yaml
spring:
data:
mongodb:
host: "mongoserver1.example.com"
port: 27017
additional-hosts:
- "mongoserver2.example.com:23456"
database: "test"
username: "user"
password: "secret"
The auto-configured MongoClient can be configured to use SSL for communication with the server
by setting the properties as shown in this example:
350
Properties
spring.data.mongodb.uri=mongodb://user:secret@mongoserver1.example.com:27017,mongoserv
er2.example.com:23456/test
spring.data.mongodb.ssl.enabled=true
Yaml
spring:
data:
mongodb:
uri:
"mongodb://user:secret@mongoserver1.example.com:27017,mongoserver2.example.com:23456/t
est"
ssl:
enabled: true
Custom SSL trust material can be configured in an SSL bundle and applied to the MongoClient as
shown in this example:
Properties
spring.data.mongodb.uri=mongodb://user:secret@mongoserver1.example.com:27017,mongoserv
er2.example.com:23456/test
spring.data.mongodb.ssl.bundle=example
Yaml
spring:
data:
mongodb:
uri:
"mongodb://user:secret@mongoserver1.example.com:27017,mongoserver2.example.com:23456/t
est"
ssl:
bundle: "example"
TIP
You can also specify the port as part of the host address by using the host:port syntax.
This format should be used if you need to change the port of an additional-hosts
entry.
351
If you do not use Spring Data MongoDB, you can inject a MongoClient bean instead of
using MongoDatabaseFactory. If you want to take complete control of establishing the
TIP
MongoDB connection, you can also declare your own MongoDatabaseFactory or
MongoClient bean.
If you are using the reactive driver, Netty is required for SSL. The auto-
NOTE configuration configures this factory automatically if Netty is available and the
factory to use has not been customized already.
MongoTemplate
Spring Data MongoDB provides a MongoTemplate class that is very similar in its design to Spring’s
JdbcTemplate. As with JdbcTemplate, Spring Boot auto-configures a bean for you to inject the
template, as follows:
Java
import com.mongodb.client.MongoCollection;
import org.bson.Document;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
352
Kotlin
import com.mongodb.client.MongoCollection
import org.bson.Document
import org.springframework.data.mongodb.core.MongoTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val mongoTemplate: MongoTemplate) {
Spring Data includes repository support for MongoDB. As with the JPA repositories discussed
earlier, the basic principle is that queries are constructed automatically, based on method names.
In fact, both Spring Data JPA and Spring Data MongoDB share the same common infrastructure. You
could take the JPA example from earlier and, assuming that City is now a MongoDB data class
rather than a JPA @Entity, it works in the same way, as shown in the following example:
Java
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.repository.Repository;
353
Kotlin
import org.springframework.data.domain.Page
import org.springframework.data.domain.Pageable
import org.springframework.data.repository.Repository
interface CityRepository :
Repository<City?, Long?> {
fun findAll(pageable: Pageable?): Page<City?>?
fun findByNameAndStateAllIgnoringCase(name: String?, state: String?): City?
}
TIP You can customize document scanning locations by using the @EntityScan annotation.
For complete details of Spring Data MongoDB, including its rich object mapping
TIP
technologies, see its reference documentation.
9.2.3. Neo4j
Neo4j is an open-source NoSQL graph database that uses a rich data model of nodes connected by
first class relationships, which is better suited for connected big data than traditional RDBMS
approaches. Spring Boot offers several conveniences for working with Neo4j, including the spring-
boot-starter-data-neo4j “Starter”.
To access a Neo4j server, you can inject an auto-configured org.neo4j.driver.Driver. By default, the
instance tries to connect to a Neo4j server at localhost:7687 using the Bolt protocol. The following
example shows how to inject a Neo4j Driver that gives you access, amongst other things, to a
Session:
354
Java
import org.neo4j.driver.Driver;
import org.neo4j.driver.Session;
import org.neo4j.driver.Values;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
355
Kotlin
import org.neo4j.driver.*
import org.springframework.stereotype.Component
@Component
class MyBean(private val driver: Driver) {
fun someMethod(message: String?): String {
driver.session().use { session ->
return@someMethod session.executeWrite { transaction: TransactionContext
->
transaction
.run(
"CREATE (a:Greeting) SET a.message = \$message RETURN
a.message + ', from node ' + id(a)",
Values.parameters("message", message)
)
.single()[0].asString()
}
}
}
}
You can configure various aspects of the driver using spring.neo4j.* properties. The following
example shows how to configure the uri and credentials to use:
Properties
spring.neo4j.uri=bolt://my-server:7687
spring.neo4j.authentication.username=neo4j
spring.neo4j.authentication.password=secret
Yaml
spring:
neo4j:
uri: "bolt://my-server:7687"
authentication:
username: "neo4j"
password: "secret"
The auto-configured Driver is created using ConfigBuilder. To fine-tune its configuration, declare
one or more ConfigBuilderCustomizer beans. Each will be called in order with the ConfigBuilder that
is used to build the Driver.
Spring Data includes repository support for Neo4j. For complete details of Spring Data Neo4j, see
the reference documentation.
356
Spring Data Neo4j shares the common infrastructure with Spring Data JPA as many other Spring
Data modules do. You could take the JPA example from earlier and define City as Spring Data Neo4j
@Node rather than JPA @Entity and the repository abstraction works in the same way, as shown in
the following example:
Java
import java.util.Optional;
import org.springframework.data.neo4j.repository.Neo4jRepository;
Kotlin
import org.springframework.data.neo4j.repository.Neo4jRepository
import java.util.Optional
You can customize the locations to look for repositories and entities by using
@EnableNeo4jRepositories and @EntityScan respectively on a @Configuration-bean.
NOTE
357
Java
import org.neo4j.driver.Driver;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import
org.springframework.data.neo4j.core.ReactiveDatabaseSelectionProvider;
import
org.springframework.data.neo4j.core.transaction.ReactiveNeo4jTransaction
Manager;
@Configuration(proxyBeanMethods = false)
public class MyNeo4jConfiguration {
@Bean
public ReactiveNeo4jTransactionManager
reactiveTransactionManager(Driver driver,
ReactiveDatabaseSelectionProvider databaseNameProvider) {
return new ReactiveNeo4jTransactionManager(driver,
databaseNameProvider);
}
Kotlin
import org.neo4j.driver.Driver
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import
org.springframework.data.neo4j.core.ReactiveDatabaseSelectionProvider
import
org.springframework.data.neo4j.core.transaction.ReactiveNeo4jTransaction
Manager
@Configuration(proxyBeanMethods = false)
class MyNeo4jConfiguration {
@Bean
fun reactiveTransactionManager(driver: Driver,
databaseNameProvider: ReactiveDatabaseSelectionProvider):
ReactiveNeo4jTransactionManager {
return ReactiveNeo4jTransactionManager(driver,
databaseNameProvider)
}
}
358
9.2.4. Elasticsearch
Elasticsearch is an open source, distributed, RESTful search and analytics engine. Spring Boot offers
basic auto-configuration for Elasticsearch clients.
Elasticsearch ships two different REST clients that you can use to query a cluster: the low-level
client from the org.elasticsearch.client:elasticsearch-rest-client module and the Java API client
from the co.elastic.clients:elasticsearch-java module. Additionally, Spring Boot provides support
for a reactive client from the org.springframework.data:spring-data-elasticsearch module. By
default, the clients will target localhost:9200. You can use spring.elasticsearch.* properties to
further tune how the clients are configured, as shown in the following example:
Properties
spring.elasticsearch.uris=https://search.example.com:9200
spring.elasticsearch.socket-timeout=10s
spring.elasticsearch.username=user
spring.elasticsearch.password=secret
Yaml
spring:
elasticsearch:
uris: "https://search.example.com:9200"
socket-timeout: "10s"
username: "user"
password: "secret"
If you have elasticsearch-rest-client on the classpath, Spring Boot will auto-configure and register
a RestClient bean. In addition to the properties described previously, to fine-tune the RestClient
you can register an arbitrary number of beans that implement RestClientBuilderCustomizer for
more advanced customizations. To take full control over the clients' configuration, define a
RestClientBuilder bean.
359
example:
Properties
spring.elasticsearch.restclient.sniffer.interval=10m
spring.elasticsearch.restclient.sniffer.delay-after-failure=30s
Yaml
spring:
elasticsearch:
restclient:
sniffer:
interval: "10m"
delay-after-failure: "30s"
The ElasticsearchClient uses a transport that depends upon the previously described RestClient.
Therefore, the properties described previously can be used to configure the ElasticsearchClient.
Furthermore, you can define a TransportOptions bean to take further control of the behavior of the
transport.
The ReactiveElasticsearchclient uses a transport that depends upon the previously described
RestClient. Therefore, the properties described previously can be used to configure the
ReactiveElasticsearchClient. Furthermore, you can define a TransportOptions bean to take further
control of the behavior of the transport.
360
Java
import org.springframework.data.elasticsearch.client.elc.ElasticsearchTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.stereotype.Component
@Component
class MyBean(private val template:
org.springframework.data.elasticsearch.client.erhlc.ElasticsearchRestTemplate ) {
In the presence of spring-data-elasticsearch and Reactor, Spring Boot can also auto-configure a
ReactiveElasticsearchClient and a ReactiveElasticsearchTemplate as beans. They are the reactive
equivalent of the other REST clients.
Spring Data includes repository support for Elasticsearch. As with the JPA repositories discussed
earlier, the basic principle is that queries are constructed for you automatically based on method
names.
In fact, both Spring Data JPA and Spring Data Elasticsearch share the same common infrastructure.
You could take the JPA example from earlier and, assuming that City is now an Elasticsearch
@Document class rather than a JPA @Entity, it works in the same way.
TIP For complete details of Spring Data Elasticsearch, see the reference documentation.
361
Spring Boot supports both classic and reactive Elasticsearch repositories, using the
ElasticsearchRestTemplate or ReactiveElasticsearchTemplate beans. Most likely those beans are
auto-configured by Spring Boot given the required dependencies are present.
If you wish to use your own template for backing the Elasticsearch repositories, you can add your
own ElasticsearchRestTemplate or ElasticsearchOperations @Bean, as long as it is named
"elasticsearchTemplate". Same applies to ReactiveElasticsearchTemplate and
ReactiveElasticsearchOperations, with the bean name "reactiveElasticsearchTemplate".
You can choose to disable the repositories support with the following property:
Properties
spring.data.elasticsearch.repositories.enabled=false
Yaml
spring:
data:
elasticsearch:
repositories:
enabled: false
9.2.5. Cassandra
Cassandra is an open source, distributed database management system designed to handle large
amounts of data across many commodity servers. Spring Boot offers auto-configuration for
Cassandra and the abstractions on top of it provided by Spring Data Cassandra. There is a spring-
boot-starter-data-cassandra “Starter” for collecting the dependencies in a convenient way.
Connecting to Cassandra
Properties
spring.cassandra.keyspace-name=mykeyspace
spring.cassandra.contact-points=cassandrahost1:9042,cassandrahost2:9042
spring.cassandra.local-datacenter=datacenter1
362
Yaml
spring:
cassandra:
keyspace-name: "mykeyspace"
contact-points: "cassandrahost1:9042,cassandrahost2:9042"
local-datacenter: "datacenter1"
If the port is the same for all your contact points you can use a shortcut and only specify the host
names, as shown in the following example:
Properties
spring.cassandra.keyspace-name=mykeyspace
spring.cassandra.contact-points=cassandrahost1,cassandrahost2
spring.cassandra.local-datacenter=datacenter1
Yaml
spring:
cassandra:
keyspace-name: "mykeyspace"
contact-points: "cassandrahost1,cassandrahost2"
local-datacenter: "datacenter1"
Those two examples are identical as the port default to 9042. If you need to configure
TIP
the port, use spring.cassandra.port.
The auto-configured CqlSession can be configured to use SSL for communication with the server by
setting the properties as shown in this example:
Properties
spring.cassandra.keyspace-name=mykeyspace
spring.cassandra.contact-points=cassandrahost1,cassandrahost2
spring.cassandra.local-datacenter=datacenter1
spring.cassandra.ssl.enabled=true
Yaml
spring:
cassandra:
keyspace-name: "mykeyspace"
contact-points: "cassandrahost1,cassandrahost2"
local-datacenter: "datacenter1"
ssl:
enabled: true
363
Custom SSL trust material can be configured in an SSL bundle and applied to the CqlSession as
shown in this example:
Properties
spring.cassandra.keyspace-name=mykeyspace
spring.cassandra.contact-points=cassandrahost1,cassandrahost2
spring.cassandra.local-datacenter=datacenter1
spring.cassandra.ssl.bundle=example
Yaml
spring:
cassandra:
keyspace-name: "mykeyspace"
contact-points: "cassandrahost1,cassandrahost2"
local-datacenter: "datacenter1"
ssl:
bundle: "example"
The Cassandra driver has its own configuration infrastructure that loads an
application.conf at the root of the classpath.
Spring Boot does not look for such a file by default but can load one using
spring.cassandra.config. If a property is both present in spring.cassandra.* and the
NOTE
configuration file, the value in spring.cassandra.* takes precedence.
For more advanced driver customizations, you can register an arbitrary number of
beans that implement DriverConfigLoaderBuilderCustomizer. The CqlSession can be
customized with a bean of type CqlSessionBuilderCustomizer.
If you use CqlSessionBuilder to create multiple CqlSession beans, keep in mind the
NOTE
builder is mutable so make sure to inject a fresh copy for each session.
364
Java
import org.springframework.data.cassandra.core.CassandraTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.data.cassandra.core.CassandraTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val template: CassandraTemplate) {
If you add your own @Bean of type CassandraTemplate, it replaces the default.
Spring Data includes basic repository support for Cassandra. Currently, this is more limited than
the JPA repositories discussed earlier and needs @Query annotated finder methods.
TIP For complete details of Spring Data Cassandra, see the reference documentation.
9.2.6. Couchbase
365
dependencies in a convenient way.
Connecting to Couchbase
You can get a Cluster by adding the Couchbase SDK and some configuration. The spring.couchbase.*
properties can be used to customize the connection. Generally, you provide the connection string,
username, and password, as shown in the following example:
Properties
spring.couchbase.connection-string=couchbase://192.168.1.123
spring.couchbase.username=user
spring.couchbase.password=secret
Yaml
spring:
couchbase:
connection-string: "couchbase://192.168.1.123"
username: "user"
password: "secret"
It is also possible to customize some of the ClusterEnvironment settings. For instance, the following
configuration changes the timeout to open a new Bucket and enables SSL support with a reference
to a configured SSL bundle:
Properties
spring.couchbase.env.timeouts.connect=3s
spring.couchbase.env.ssl.bundle=example
Yaml
spring:
couchbase:
env:
timeouts:
connect: "3s"
ssl:
bundle: "example"
Check the spring.couchbase.env.* properties for more details. To take more control,
TIP
one or more ClusterEnvironmentBuilderCustomizer beans can be used.
Spring Data includes repository support for Couchbase. For complete details of Spring Data
Couchbase, see the reference documentation.
366
You can inject an auto-configured CouchbaseTemplate instance as you would with any other Spring
Bean, provided a CouchbaseClientFactory bean is available. This happens when a Cluster is
available, as described above, and a bucket name has been specified:
Properties
spring.data.couchbase.bucket-name=my-bucket
Yaml
spring:
data:
couchbase:
bucket-name: "my-bucket"
Java
import org.springframework.data.couchbase.core.CouchbaseTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
367
Kotlin
import org.springframework.data.couchbase.core.CouchbaseTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val template: CouchbaseTemplate) {
There are a few beans that you can define in your own configuration to override those provided by
the auto-configuration:
To avoid hard-coding those names in your own config, you can reuse BeanNames provided by Spring
Data Couchbase. For instance, you can customize the converters to use, as follows:
Java
import org.assertj.core.util.Arrays;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.couchbase.config.BeanNames;
import org.springframework.data.couchbase.core.convert.CouchbaseCustomConversions;
@Configuration(proxyBeanMethods = false)
public class MyCouchbaseConfiguration {
@Bean(BeanNames.COUCHBASE_CUSTOM_CONVERSIONS)
public CouchbaseCustomConversions myCustomConversions() {
return new CouchbaseCustomConversions(Arrays.asList(new MyConverter()));
}
368
Kotlin
import org.assertj.core.util.Arrays
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.data.couchbase.config.BeanNames
import org.springframework.data.couchbase.core.convert.CouchbaseCustomConversions
@Configuration(proxyBeanMethods = false)
class MyCouchbaseConfiguration {
@Bean(BeanNames.COUCHBASE_CUSTOM_CONVERSIONS)
fun myCustomConversions(): CouchbaseCustomConversions {
return CouchbaseCustomConversions(Arrays.asList(MyConverter()))
}
9.2.7. LDAP
To connect to an LDAP server, make sure you declare a dependency on the spring-boot-starter-
data-ldap “Starter” or spring-ldap-core and then declare the URLs of your server in your
application.properties, as shown in the following example:
Properties
spring.ldap.urls=ldap://myserver:1235
spring.ldap.username=admin
spring.ldap.password=secret
Yaml
spring:
ldap:
urls: "ldap://myserver:1235"
username: "admin"
password: "secret"
If you need to customize connection settings, you can use the spring.ldap.base and
369
spring.ldap.base-environment properties.
Spring Data includes repository support for LDAP. For complete details of Spring Data LDAP, see the
reference documentation.
You can also inject an auto-configured LdapTemplate instance as you would with any other Spring
Bean, as shown in the following example:
Java
import java.util.List;
import org.springframework.ldap.core.LdapTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
370
Kotlin
import org.springframework.ldap.core.LdapTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val template: LdapTemplate) {
For testing purposes, Spring Boot supports auto-configuration of an in-memory LDAP server from
UnboundID. To configure the server, add a dependency to com.unboundid:unboundid-ldapsdk and
declare a spring.ldap.embedded.base-dn property, as follows:
Properties
spring.ldap.embedded.base-dn=dc=spring,dc=io
Yaml
spring:
ldap:
embedded:
base-dn: "dc=spring,dc=io"
In yaml files, you can use the yaml list notation. In properties files, you must
include the index as part of the property name:
Properties
NOTE spring.ldap.embedded.base-dn[0]=dc=spring,dc=io
spring.ldap.embedded.base-dn[1]=dc=vmware,dc=com
Yaml
spring.ldap.embedded.base-dn:
- "dc=spring,dc=io"
- "dc=vmware,dc=com"
371
By default, the server starts on a random port and triggers the regular LDAP support. There is no
need to specify a spring.ldap.urls property.
If there is a schema.ldif file on your classpath, it is used to initialize the server. If you want to load
the initialization script from a different resource, you can also use the spring.ldap.embedded.ldif
property.
By default, a standard schema is used to validate LDIF files. You can turn off validation altogether by
setting the spring.ldap.embedded.validation.enabled property. If you have custom attributes, you
can use spring.ldap.embedded.validation.schema to define your custom attribute types or object
classes.
9.2.8. InfluxDB
InfluxDB is an open-source time series database optimized for fast, high-availability storage and
retrieval of time series data in fields such as operations monitoring, application metrics, Internet-
of-Things sensor data, and real-time analytics.
Connecting to InfluxDB
Spring Boot auto-configures an InfluxDB instance, provided the influxdb-java client is on the
classpath and the URL of the database is set, as shown in the following example:
Properties
spring.influx.url=https://172.0.0.1:8086
Yaml
spring:
influx:
url: "https://172.0.0.1:8086"
If the connection to InfluxDB requires a user and password, you can set the spring.influx.user and
spring.influx.password properties accordingly.
InfluxDB relies on OkHttp. If you need to tune the http client InfluxDB uses behind the scenes, you
can register an InfluxDbOkHttpClientBuilderProvider bean.
If you need more control over the configuration, consider registering an InfluxDbCustomizer bean.
372
Chapter 10. Messaging
The Spring Framework provides extensive support for integrating with messaging systems, from
simplified use of the JMS API using JmsTemplate to a complete infrastructure to receive messages
asynchronously. Spring AMQP provides a similar feature set for the Advanced Message Queuing
Protocol. Spring Boot also provides auto-configuration options for RabbitTemplate and RabbitMQ.
Spring WebSocket natively includes support for STOMP messaging, and Spring Boot has support for
that through starters and a small amount of auto-configuration. Spring Boot also has support for
Apache Kafka.
10.1. JMS
The jakarta.jms.ConnectionFactory interface provides a standard method of creating a
jakarta.jms.Connection for interacting with a JMS broker. Although Spring needs a
ConnectionFactory to work with JMS, you generally need not use it directly yourself and can instead
rely on higher level messaging abstractions. (See the relevant section of the Spring Framework
reference documentation for details.) Spring Boot also auto-configures the necessary infrastructure
to send and receive messages.
When ActiveMQ is available on the classpath, Spring Boot can configure a ConnectionFactory.
Properties
spring.activemq.broker-url=tcp://192.168.1.210:9876
spring.activemq.user=admin
spring.activemq.password=secret
Yaml
spring:
activemq:
broker-url: "tcp://192.168.1.210:9876"
user: "admin"
password: "secret"
373
Properties
spring.jms.cache.session-cache-size=5
Yaml
spring:
jms:
cache:
session-cache-size: 5
Properties
spring.activemq.pool.enabled=true
spring.activemq.pool.max-connections=50
Yaml
spring:
activemq:
pool:
enabled: true
max-connections: 50
See ActiveMQProperties for more of the supported options. You can also register an
TIP arbitrary number of beans that implement ActiveMQConnectionFactoryCustomizer for
more advanced customizations.
By default, ActiveMQ creates a destination if it does not yet exist so that destinations are resolved
against their provided names.
Spring Boot can auto-configure a ConnectionFactory when it detects that ActiveMQ Artemis is
available on the classpath. If the broker is present, an embedded broker is automatically started
and configured (unless the mode property has been explicitly set). The supported modes are
embedded (to make explicit that an embedded broker is required and that an error should occur if
the broker is not available on the classpath) and native (to connect to a broker using the netty
transport protocol). When the latter is configured, Spring Boot configures a ConnectionFactory that
connects to a broker running on the local machine with the default settings.
374
If you use spring-boot-starter-artemis, the necessary dependencies to connect to an
existing ActiveMQ Artemis instance are provided, as well as the Spring
NOTE
infrastructure to integrate with JMS. Adding org.apache.activemq:artemis-jakarta-
server to your application lets you use embedded mode.
Properties
spring.artemis.mode=native
spring.artemis.broker-url=tcp://192.168.1.210:9876
spring.artemis.user=admin
spring.artemis.password=secret
Yaml
spring:
artemis:
mode: native
broker-url: "tcp://192.168.1.210:9876"
user: "admin"
password: "secret"
When embedding the broker, you can choose if you want to enable persistence and list the
destinations that should be made available. These can be specified as a comma-separated list to
create them with the default options, or you can define bean(s) of type
org.apache.activemq.artemis.jms.server.config.JMSQueueConfiguration or
org.apache.activemq.artemis.jms.server.config.TopicConfiguration, for advanced queue and topic
configurations, respectively.
Properties
spring.jms.cache.session-cache-size=5
Yaml
spring:
jms:
cache:
session-cache-size: 5
375
Properties
spring.artemis.pool.enabled=true
spring.artemis.pool.max-connections=50
Yaml
spring:
artemis:
pool:
enabled: true
max-connections: 50
No JNDI lookup is involved, and destinations are resolved against their names, using either the name
attribute in the Artemis configuration or the names provided through configuration.
If you are running your application in an application server, Spring Boot tries to locate a JMS
ConnectionFactory by using JNDI. By default, the java:/JmsXA and java:/XAConnectionFactory location
are checked. You can use the spring.jms.jndi-name property if you need to specify an alternative
location, as shown in the following example:
Properties
spring.jms.jndi-name=java:/MyConnectionFactory
Yaml
spring:
jms:
jndi-name: "java:/MyConnectionFactory"
Spring’s JmsTemplate is auto-configured, and you can autowire it directly into your own beans, as
shown in the following example:
376
Java
import org.springframework.jms.core.JmsTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.jms.core.JmsTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val jmsTemplate: JmsTemplate) {
fun someMethod() {
jmsTemplate.convertAndSend("hello")
}
When the JMS infrastructure is present, any bean can be annotated with @JmsListener to create a
listener endpoint. If no JmsListenerContainerFactory has been defined, a default one is configured
automatically. If a DestinationResolver, a MessageConverter, or a jakarta.jms.ExceptionListener
beans are defined, they are associated automatically with the default factory.
377
method (or a delegate thereof). This ensures that the incoming message is acknowledged, once the
local transaction has completed. This also includes sending response messages that have been
performed on the same JMS session.
Java
import org.springframework.jms.annotation.JmsListener;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
@JmsListener(destination = "someQueue")
public void processMessage(String content) {
// ...
}
Kotlin
import org.springframework.jms.annotation.JmsListener
import org.springframework.stereotype.Component
@Component
class MyBean {
@JmsListener(destination = "someQueue")
fun processMessage(content: String?) {
// ...
}
If you need to create more JmsListenerContainerFactory instances or if you want to override the
default, Spring Boot provides a DefaultJmsListenerContainerFactoryConfigurer that you can use to
initialize a DefaultJmsListenerContainerFactory with the same settings as the one that is auto-
configured.
For instance, the following example exposes another factory that uses a specific MessageConverter:
378
Java
import jakarta.jms.ConnectionFactory;
import
org.springframework.boot.autoconfigure.jms.DefaultJmsListenerContainerFactoryConfigure
r;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jms.config.DefaultJmsListenerContainerFactory;
@Configuration(proxyBeanMethods = false)
public class MyJmsConfiguration {
@Bean
public DefaultJmsListenerContainerFactory
myFactory(DefaultJmsListenerContainerFactoryConfigurer configurer) {
DefaultJmsListenerContainerFactory factory = new
DefaultJmsListenerContainerFactory();
ConnectionFactory connectionFactory = getCustomConnectionFactory();
configurer.configure(factory, connectionFactory);
factory.setMessageConverter(new MyMessageConverter());
return factory;
}
379
Kotlin
import jakarta.jms.ConnectionFactory
import
org.springframework.boot.autoconfigure.jms.DefaultJmsListenerContainerFactoryConfigure
r
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.jms.config.DefaultJmsListenerContainerFactory
@Configuration(proxyBeanMethods = false)
class MyJmsConfiguration {
@Bean
fun myFactory(configurer: DefaultJmsListenerContainerFactoryConfigurer):
DefaultJmsListenerContainerFactory {
val factory = DefaultJmsListenerContainerFactory()
val connectionFactory = getCustomConnectionFactory()
configurer.configure(factory, connectionFactory)
factory.setMessageConverter(MyMessageConverter())
return factory
}
Then you can use the factory in any @JmsListener-annotated method as follows:
Java
import org.springframework.jms.annotation.JmsListener;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
380
Kotlin
import org.springframework.jms.annotation.JmsListener
import org.springframework.stereotype.Component
@Component
class MyBean {
10.2. AMQP
The Advanced Message Queuing Protocol (AMQP) is a platform-neutral, wire-level protocol for
message-oriented middleware. The Spring AMQP project applies core Spring concepts to the
development of AMQP-based messaging solutions. Spring Boot offers several conveniences for
working with AMQP through RabbitMQ, including the spring-boot-starter-amqp “Starter”.
RabbitMQ is a lightweight, reliable, scalable, and portable message broker based on the AMQP
protocol. Spring uses RabbitMQ to communicate through the AMQP protocol.
Properties
spring.rabbitmq.host=localhost
spring.rabbitmq.port=5672
spring.rabbitmq.username=admin
spring.rabbitmq.password=secret
Yaml
spring:
rabbitmq:
host: "localhost"
port: 5672
username: "admin"
password: "secret"
Alternatively, you could configure the same connection using the addresses attribute:
381
Properties
spring.rabbitmq.addresses=amqp://admin:secret@localhost
Yaml
spring:
rabbitmq:
addresses: "amqp://admin:secret@localhost"
When specifying addresses that way, the host and port properties are ignored. If the
NOTE
address uses the amqps protocol, SSL support is enabled automatically.
See RabbitProperties for more of the supported property-based configuration options. To configure
lower-level details of the RabbitMQ ConnectionFactory that is used by Spring AMQP, define a
ConnectionFactoryCustomizer bean.
TIP See Understanding AMQP, the protocol used by RabbitMQ for more details.
Spring’s AmqpTemplate and AmqpAdmin are auto-configured, and you can autowire them directly into
your own beans, as shown in the following example:
382
Java
import org.springframework.amqp.core.AmqpAdmin;
import org.springframework.amqp.core.AmqpTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.amqp.core.AmqpAdmin
import org.springframework.amqp.core.AmqpTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val amqpAdmin: AmqpAdmin, private val amqpTemplate: AmqpTemplate)
{
fun someMethod() {
amqpAdmin.getQueueInfo("someQueue")
}
fun someOtherMethod() {
amqpTemplate.convertAndSend("hello")
}
383
RabbitMessagingTemplate can be injected in a similar manner. If a MessageConverter
NOTE
bean is defined, it is associated automatically to the auto-configured AmqpTemplate.
To retry operations, you can enable retries on the AmqpTemplate (for example, in the event that the
broker connection is lost):
Properties
spring.rabbitmq.template.retry.enabled=true
spring.rabbitmq.template.retry.initial-interval=2s
Yaml
spring:
rabbitmq:
template:
retry:
enabled: true
initial-interval: "2s"
Retries are disabled by default. You can also customize the RetryTemplate programmatically by
declaring a RabbitRetryTemplateCustomizer bean.
If you need to create more RabbitTemplate instances or if you want to override the default, Spring
Boot provides a RabbitTemplateConfigurer bean that you can use to initialize a RabbitTemplate with
the same settings as the factories used by the auto-configuration.
To send a message to a particular stream, specify the name of the stream, as shown in the following
example:
Properties
spring.rabbitmq.stream.name=my-stream
Yaml
spring:
rabbitmq:
stream:
name: "my-stream"
384
If you need to create more RabbitStreamTemplate instances or if you want to override the default,
Spring Boot provides a RabbitStreamTemplateConfigurer bean that you can use to initialize a
RabbitStreamTemplate with the same settings as the factories used by the auto-configuration.
When the Rabbit infrastructure is present, any bean can be annotated with @RabbitListener to
create a listener endpoint. If no RabbitListenerContainerFactory has been defined, a default
SimpleRabbitListenerContainerFactory is automatically configured and you can switch to a direct
container using the spring.rabbitmq.listener.type property. If a MessageConverter or a
MessageRecoverer bean is defined, it is automatically associated with the default factory.
The following sample component creates a listener endpoint on the someQueue queue:
Java
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
@RabbitListener(queues = "someQueue")
public void processMessage(String content) {
// ...
}
Kotlin
import org.springframework.amqp.rabbit.annotation.RabbitListener
import org.springframework.stereotype.Component
@Component
class MyBean {
@RabbitListener(queues = ["someQueue"])
fun processMessage(content: String?) {
// ...
}
If you need to create more RabbitListenerContainerFactory instances or if you want to override the
default, Spring Boot provides a SimpleRabbitListenerContainerFactoryConfigurer and a
DirectRabbitListenerContainerFactoryConfigurer that you can use to initialize a
385
SimpleRabbitListenerContainerFactory and a DirectRabbitListenerContainerFactory with the same
settings as the factories used by the auto-configuration.
It does not matter which container type you chose. Those two beans are exposed by
TIP
the auto-configuration.
For instance, the following configuration class exposes another factory that uses a specific
MessageConverter:
Java
import org.springframework.amqp.rabbit.config.SimpleRabbitListenerContainerFactory;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import
org.springframework.boot.autoconfigure.amqp.SimpleRabbitListenerContainerFactoryConfig
urer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyRabbitConfiguration {
@Bean
public SimpleRabbitListenerContainerFactory
myFactory(SimpleRabbitListenerContainerFactoryConfigurer configurer) {
SimpleRabbitListenerContainerFactory factory = new
SimpleRabbitListenerContainerFactory();
ConnectionFactory connectionFactory = getCustomConnectionFactory();
configurer.configure(factory, connectionFactory);
factory.setMessageConverter(new MyMessageConverter());
return factory;
}
386
Kotlin
import org.springframework.amqp.rabbit.config.SimpleRabbitListenerContainerFactory
import org.springframework.amqp.rabbit.connection.ConnectionFactory
import
org.springframework.boot.autoconfigure.amqp.SimpleRabbitListenerContainerFactoryConfig
urer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyRabbitConfiguration {
@Bean
fun myFactory(configurer: SimpleRabbitListenerContainerFactoryConfigurer):
SimpleRabbitListenerContainerFactory {
val factory = SimpleRabbitListenerContainerFactory()
val connectionFactory = getCustomConnectionFactory()
configurer.configure(factory, connectionFactory)
factory.setMessageConverter(MyMessageConverter())
return factory
}
Then you can use the factory in any @RabbitListener-annotated method, as follows:
Java
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
387
Kotlin
import org.springframework.amqp.rabbit.annotation.RabbitListener
import org.springframework.stereotype.Component
@Component
class MyBean {
You can enable retries to handle situations where your listener throws an exception. By default,
RejectAndDontRequeueRecoverer is used, but you can define a MessageRecoverer of your own. When
retries are exhausted, the message is rejected and either dropped or routed to a dead-letter
exchange if the broker is configured to do so. By default, retries are disabled. You can also
customize the RetryTemplate programmatically by declaring a RabbitRetryTemplateCustomizer bean.
By default, if retries are disabled and the listener throws an exception, the
delivery is retried indefinitely. You can modify this behavior in two ways: Set
the defaultRequeueRejected property to false so that zero re-deliveries are
IMPORTANT
attempted or throw an AmqpRejectAndDontRequeueException to signal the
message should be rejected. The latter is the mechanism used when retries
are enabled and the maximum number of delivery attempts is reached.
Properties
spring.kafka.bootstrap-servers=localhost:9092
spring.kafka.consumer.group-id=myGroup
Yaml
spring:
kafka:
bootstrap-servers: "localhost:9092"
consumer:
group-id: "myGroup"
388
To create a topic on startup, add a bean of type NewTopic. If the topic already exists, the
TIP
bean is ignored.
Spring’s KafkaTemplate is auto-configured, and you can autowire it directly in your own beans, as
shown in the following example:
Java
import org.springframework.kafka.core.KafkaTemplate;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
Kotlin
import org.springframework.kafka.core.KafkaTemplate
import org.springframework.stereotype.Component
@Component
class MyBean(private val kafkaTemplate: KafkaTemplate<String, String>) {
fun someMethod() {
kafkaTemplate.send("someTopic", "Hello")
}
389
10.3.2. Receiving a Message
When the Apache Kafka infrastructure is present, any bean can be annotated with @KafkaListener
to create a listener endpoint. If no KafkaListenerContainerFactory has been defined, a default one is
automatically configured with keys defined in spring.kafka.listener.*.
Java
import org.springframework.kafka.annotation.KafkaListener;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
@KafkaListener(topics = "someTopic")
public void processMessage(String content) {
// ...
}
Kotlin
import org.springframework.kafka.annotation.KafkaListener
import org.springframework.stereotype.Component
@Component
class MyBean {
@KafkaListener(topics = ["someTopic"])
fun processMessage(content: String?) {
// ...
}
390
10.3.3. Kafka Streams
Spring for Apache Kafka provides a factory bean to create a StreamsBuilder object and manage the
lifecycle of its streams. Spring Boot auto-configures the required KafkaStreamsConfiguration bean as
long as kafka-streams is on the classpath and Kafka Streams is enabled by the @EnableKafkaStreams
annotation.
Enabling Kafka Streams means that the application id and bootstrap servers must be set. The
former can be configured using spring.kafka.streams.application-id, defaulting to
spring.application.name if not set. The latter can be set globally or specifically overridden only for
streams.
Several additional properties are available using dedicated properties; other arbitrary Kafka
properties can be set using the spring.kafka.streams.properties namespace. See also Additional
Kafka Properties for more information.
To use the factory bean, wire StreamsBuilder into your @Bean as shown in the following example:
Java
import org.apache.kafka.common.serialization.Serdes;
import org.apache.kafka.streams.KeyValue;
import org.apache.kafka.streams.StreamsBuilder;
import org.apache.kafka.streams.kstream.KStream;
import org.apache.kafka.streams.kstream.Produced;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.kafka.annotation.EnableKafkaStreams;
import org.springframework.kafka.support.serializer.JsonSerde;
@Configuration(proxyBeanMethods = false)
@EnableKafkaStreams
public class MyKafkaStreamsConfiguration {
@Bean
public KStream<Integer, String> kStream(StreamsBuilder streamsBuilder) {
KStream<Integer, String> stream = streamsBuilder.stream("ks1In");
stream.map(this::uppercaseValue).to("ks1Out", Produced.with(Serdes.Integer(),
new JsonSerde<>()));
return stream;
}
391
Kotlin
import org.apache.kafka.common.serialization.Serdes
import org.apache.kafka.streams.KeyValue
import org.apache.kafka.streams.StreamsBuilder
import org.apache.kafka.streams.kstream.KStream
import org.apache.kafka.streams.kstream.Produced
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.kafka.annotation.EnableKafkaStreams
import org.springframework.kafka.support.serializer.JsonSerde
@Configuration(proxyBeanMethods = false)
@EnableKafkaStreams
class MyKafkaStreamsConfiguration {
@Bean
fun kStream(streamsBuilder: StreamsBuilder): KStream<Int, String> {
val stream = streamsBuilder.stream<Int, String>("ks1In")
stream.map(this::uppercaseValue).to("ks1Out", Produced.with(Serdes.Integer(),
JsonSerde()))
return stream
}
By default, the streams managed by the StreamBuilder object are started automatically. You can
customize this behavior using the spring.kafka.streams.auto-startup property.
The properties supported by auto configuration are shown in the “Integration Properties” section of
the Appendix. Note that, for the most part, these properties (hyphenated or camelCase) map
directly to the Apache Kafka dotted properties. See the Apache Kafka documentation for details.
The first few of these properties apply to all components (producers, consumers, admins, and
streams) but can be specified at the component level if you wish to use different values. Apache
Kafka designates properties with an importance of HIGH, MEDIUM, or LOW. Spring Boot auto-
configuration supports all HIGH importance properties, some selected MEDIUM and LOW
properties, and any properties that do not have a default value.
Only a subset of the properties supported by Kafka are available directly through the
KafkaProperties class. If you wish to configure the producer or consumer with additional properties
that are not directly supported, use the following properties:
392
Properties
spring.kafka.properties[prop.one]=first
spring.kafka.admin.properties[prop.two]=second
spring.kafka.consumer.properties[prop.three]=third
spring.kafka.producer.properties[prop.four]=fourth
spring.kafka.streams.properties[prop.five]=fifth
Yaml
spring:
kafka:
properties:
"[prop.one]": "first"
admin:
properties:
"[prop.two]": "second"
consumer:
properties:
"[prop.three]": "third"
producer:
properties:
"[prop.four]": "fourth"
streams:
properties:
"[prop.five]": "fifth"
This sets the common prop.one Kafka property to first (applies to producers, consumers and
admins), the prop.two admin property to second, the prop.three consumer property to third, the
prop.four producer property to fourth and the prop.five streams property to fifth.
Properties
spring.kafka.consumer.value-
deserializer=org.springframework.kafka.support.serializer.JsonDeserializer
spring.kafka.consumer.properties[spring.json.value.default.type]=com.example.Invoice
spring.kafka.consumer.properties[spring.json.trusted.packages]=com.example.main,com.ex
ample.another
393
Yaml
spring:
kafka:
consumer:
value-deserializer:
"org.springframework.kafka.support.serializer.JsonDeserializer"
properties:
"[spring.json.value.default.type]": "com.example.Invoice"
"[spring.json.trusted.packages]": "com.example.main,com.example.another"
Similarly, you can disable the JsonSerializer default behavior of sending type information in
headers:
Properties
spring.kafka.producer.value-
serializer=org.springframework.kafka.support.serializer.JsonSerializer
spring.kafka.producer.properties[spring.json.add.type.headers]=false
Yaml
spring:
kafka:
producer:
value-serializer: "org.springframework.kafka.support.serializer.JsonSerializer"
properties:
"[spring.json.add.type.headers]": false
Properties set in this way override any configuration item that Spring Boot
IMPORTANT
explicitly supports.
Spring for Apache Kafka provides a convenient way to test projects with an embedded Apache
Kafka broker. To use this feature, annotate a test class with @EmbeddedKafka from the spring-kafka-
test module. For more information, please see the Spring for Apache Kafka reference manual.
To make Spring Boot auto-configuration work with the aforementioned embedded Apache Kafka
broker, you need to remap a system property for embedded broker addresses (populated by the
EmbeddedKafkaBroker) into the Spring Boot configuration property for Apache Kafka. There are
several ways to do that:
394
Java
static {
System.setProperty(EmbeddedKafkaBroker.BROKER_LIST_PROPERTY,
"spring.kafka.bootstrap-servers");
}
Kotlin
init {
System.setProperty(EmbeddedKafkaBroker.BROKER_LIST_PROPERTY,
"spring.kafka.bootstrap-servers")
}
Java
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.kafka.test.context.EmbeddedKafka;
@SpringBootTest
@EmbeddedKafka(topics = "someTopic", bootstrapServersProperty =
"spring.kafka.bootstrap-servers")
class MyTest {
// ...
Kotlin
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.kafka.test.context.EmbeddedKafka
@SpringBootTest
@EmbeddedKafka(topics = ["someTopic"], bootstrapServersProperty =
"spring.kafka.bootstrap-servers")
class MyTest {
// ...
395
Properties
spring.kafka.bootstrap-servers=${spring.embedded.kafka.brokers}
Yaml
spring:
kafka:
bootstrap-servers: "${spring.embedded.kafka.brokers}"
10.4. RSocket
RSocket is a binary protocol for use on byte stream transports. It enables symmetric interaction
models through async message passing over a single connection.
The spring-messaging module of the Spring Framework provides support for RSocket requesters
and responders, both on the client and on the server side. See the RSocket section of the Spring
Framework reference for more details, including an overview of the RSocket protocol.
Spring Boot auto-configures an RSocketStrategies bean that provides all the required infrastructure
for encoding and decoding RSocket payloads. By default, the auto-configuration will try to configure
the following (in order):
The spring-boot-starter-rsocket starter provides both dependencies. See the Jackson support
section to know more about customization possibilities.
Developers can customize the RSocketStrategies component by creating beans that implement the
RSocketStrategiesCustomizer interface. Note that their @Order is important, as it determines the
order of codecs.
Spring Boot provides RSocket server auto-configuration. The required dependencies are provided
by the spring-boot-starter-rsocket.
Spring Boot allows exposing RSocket over WebSocket from a WebFlux server, or standing up an
independent RSocket server. This depends on the type of application and its configuration.
For WebFlux application (that is of type WebApplicationType.REACTIVE), the RSocket server will be
plugged into the Web Server only if the following properties match:
396
Properties
spring.rsocket.server.mapping-path=/rsocket
spring.rsocket.server.transport=websocket
Yaml
spring:
rsocket:
server:
mapping-path: "/rsocket"
transport: "websocket"
Plugging RSocket into a web server is only supported with Reactor Netty, as
WARNING
RSocket itself is built with that library.
Properties
spring.rsocket.server.port=9898
Yaml
spring:
rsocket:
server:
port: 9898
Spring Boot will auto-configure the Spring Messaging infrastructure for RSocket.
This means that Spring Boot will create a RSocketMessageHandler bean that will handle RSocket
requests to your application.
Once the RSocket channel is established between server and client, any party can send or receive
requests to the other.
As a server, you can get injected with an RSocketRequester instance on any handler method of an
RSocket @Controller. As a client, you need to configure and establish an RSocket connection first.
Spring Boot auto-configures an RSocketRequester.Builder for such cases with the expected codecs
and applies any RSocketConnectorConfigurer bean.
397
The RSocketRequester.Builder instance is a prototype bean, meaning each injection point will
provide you with a new instance . This is done on purpose since this builder is stateful and you
should not create requesters with different setups using the same instance.
Java
import reactor.core.publisher.Mono;
import org.springframework.messaging.rsocket.RSocketRequester;
import org.springframework.stereotype.Service;
@Service
public class MyService {
398
Kotlin
import org.springframework.messaging.rsocket.RSocketRequester
import org.springframework.stereotype.Service
import reactor.core.publisher.Mono
@Service
class MyService(rsocketRequesterBuilder: RSocketRequester.Builder) {
init {
rsocketRequester = rsocketRequesterBuilder.tcp("example.org", 9898)
}
Spring Integration polling logic relies on the auto-configured TaskScheduler. The default
PollerMetadata (poll unbounded number of messages every second) can be customized with
spring.integration.poller.* configuration properties.
Spring Boot also configures some features that are triggered by the presence of additional Spring
Integration modules. If spring-integration-jmx is also on the classpath, message processing
statistics are published over JMX. If spring-integration-jdbc is available, the default database
schema can be created on startup, as shown in the following line:
Properties
spring.integration.jdbc.initialize-schema=always
399
Yaml
spring:
integration:
jdbc:
initialize-schema: "always"
Properties
Yaml
Properties
Yaml
400
10.6. WebSockets
Spring Boot provides WebSockets auto-configuration for embedded Tomcat, Jetty, and Undertow. If
you deploy a war file to a standalone container, Spring Boot assumes that the container is
responsible for the configuration of its WebSocket support.
Spring Framework provides rich WebSocket support for MVC web applications that can be easily
accessed through the spring-boot-starter-websocket module.
WebSocket support is also available for reactive web applications and requires to include the
WebSocket API alongside spring-boot-starter-webflux:
<dependency>
<groupId>jakarta.websocket</groupId>
<artifactId>jakarta.websocket-api</artifactId>
</dependency>
401
Chapter 11. IO
Most applications will need to deal with input and output concerns at some point. Spring Boot
provides utilities and integrations with a range of technologies to help when you need IO
capabilities. This section covers standard IO features such as caching and validation as well as more
advanced topics such as scheduling and distributed transactions. We will also cover calling remote
REST or SOAP services and sending email.
11.1. Caching
The Spring Framework provides support for transparently adding caching to an application. At its
core, the abstraction applies caching to methods, thus reducing the number of executions based on
the information available in the cache. The caching logic is applied transparently, without any
interference to the invoker. Spring Boot auto-configures the cache infrastructure as long as caching
support is enabled by using the @EnableCaching annotation.
NOTE Check the relevant section of the Spring Framework reference for more details.
In a nutshell, to add caching to an operation of your service add the relevant annotation to its
method, as shown in the following example:
Java
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Component;
@Component
public class MyMathService {
@Cacheable("piDecimals")
public int computePiDecimal(int precision) {
...
}
402
Kotlin
import org.springframework.cache.annotation.Cacheable
import org.springframework.stereotype.Component
@Component
class MyMathService {
@Cacheable("piDecimals")
fun computePiDecimal(precision: Int): Int {
...
}
This example demonstrates the use of caching on a potentially costly operation. Before invoking
computePiDecimal, the abstraction looks for an entry in the piDecimals cache that matches the i
argument. If an entry is found, the content in the cache is immediately returned to the caller, and
the method is not invoked. Otherwise, the method is invoked, and the cache is updated before
returning the value.
You can also use the standard JSR-107 (JCache) annotations (such as
CAUTION @CacheResult) transparently. However, we strongly advise you to not mix and
match the Spring Cache and JCache annotations.
If you do not add any specific cache library, Spring Boot auto-configures a simple provider that uses
concurrent maps in memory. When a cache is required (such as piDecimals in the preceding
example), this provider creates it for you. The simple provider is not really recommended for
production usage, but it is great for getting started and making sure that you understand the
features. When you have made up your mind about the cache provider to use, please make sure to
read its documentation to figure out how to configure the caches that your application uses. Nearly
all providers require you to explicitly configure every cache that you use in the application. Some
offer a way to customize the default caches defined by the spring.cache.cache-names property.
TIP It is also possible to transparently update or evict data from the cache.
The cache abstraction does not provide an actual store and relies on abstraction materialized by the
org.springframework.cache.Cache and org.springframework.cache.CacheManager interfaces.
If you have not defined a bean of type CacheManager or a CacheResolver named cacheResolver (see
CachingConfigurer), Spring Boot tries to detect the following providers (in the indicated order):
1. Generic
3. Hazelcast
403
4. Infinispan
5. Couchbase
6. Redis
7. Caffeine
8. Cache2k
9. Simple
Additionally, Spring Boot for Apache Geode provides auto-configuration for using Apache Geode as
a cache provider.
If the CacheManager is auto-configured by Spring Boot, you can further tune its configuration before
it is fully initialized by exposing a bean that implements the CacheManagerCustomizer interface. The
following example sets a flag to say that null values should not be passed down to the underlying
map:
Java
import org.springframework.boot.autoconfigure.cache.CacheManagerCustomizer;
import org.springframework.cache.concurrent.ConcurrentMapCacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyCacheManagerConfiguration {
@Bean
public CacheManagerCustomizer<ConcurrentMapCacheManager> cacheManagerCustomizer()
{
return (cacheManager) -> cacheManager.setAllowNullValues(false);
}
404
Kotlin
import org.springframework.boot.autoconfigure.cache.CacheManagerCustomizer
import org.springframework.cache.concurrent.ConcurrentMapCacheManager
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyCacheManagerConfiguration {
@Bean
fun cacheManagerCustomizer(): CacheManagerCustomizer<ConcurrentMapCacheManager> {
return CacheManagerCustomizer { cacheManager ->
cacheManager.isAllowNullValues = false
}
}
Generic
Generic caching is used if the context defines at least one org.springframework.cache.Cache bean. A
CacheManager wrapping all beans of that type is created.
JCache (JSR-107)
It might happen that more than one provider is present, in which case the provider must be
explicitly specified. Even if the JSR-107 standard does not enforce a standardized way to define the
location of the configuration file, Spring Boot does its best to accommodate setting a cache with
implementation details, as shown in the following example:
Properties
405
Yaml
When a cache library offers both a native implementation and JSR-107 support,
NOTE Spring Boot prefers the JSR-107 support, so that the same features are available if
you switch to a different JSR-107 implementation.
Hazelcast
Spring Boot has general support for Hazelcast. If a HazelcastInstance has been auto-configured and
com.hazelcast:hazelcast-spring is on the classpath, it is automatically wrapped in a CacheManager.
Infinispan
Infinispan has no default configuration file location, so it must be specified explicitly. Otherwise,
the default bootstrap is used.
406
Properties
spring.cache.infinispan.config=infinispan.xml
Yaml
spring:
cache:
infinispan:
config: "infinispan.xml"
To be compatible with Spring Boot’s Jakarta EE 9 baseline, Infinispan’s -jakarta modules must be
used. For every module with a -jakarta variant, the variant must be used in place of the standard
module. For example, infinispan-core-jakarta and infinispan-commons-jakarta must be used in
place of infinispan-core and infinispan-commons respectively.
Couchbase
Properties
spring.cache.cache-names=cache1,cache2
spring.cache.couchbase.expiration=10m
Yaml
spring:
cache:
cache-names: "cache1,cache2"
couchbase:
expiration: "10m"
407
Java
import java.time.Duration;
import
org.springframework.boot.autoconfigure.cache.CouchbaseCacheManagerBuilderCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.couchbase.cache.CouchbaseCacheConfiguration;
@Configuration(proxyBeanMethods = false)
public class MyCouchbaseCacheManagerConfiguration {
@Bean
public CouchbaseCacheManagerBuilderCustomizer
myCouchbaseCacheManagerBuilderCustomizer() {
return (builder) -> builder
.withCacheConfiguration("cache1", CouchbaseCacheConfiguration
.defaultCacheConfig().entryExpiry(Duration.ofSeconds(10)))
.withCacheConfiguration("cache2", CouchbaseCacheConfiguration
.defaultCacheConfig().entryExpiry(Duration.ofMinutes(1)));
408
Kotlin
import
org.springframework.boot.autoconfigure.cache.CouchbaseCacheManagerBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.data.couchbase.cache.CouchbaseCacheConfiguration
import java.time.Duration
@Configuration(proxyBeanMethods = false)
class MyCouchbaseCacheManagerConfiguration {
@Bean
fun myCouchbaseCacheManagerBuilderCustomizer():
CouchbaseCacheManagerBuilderCustomizer {
return CouchbaseCacheManagerBuilderCustomizer { builder ->
builder
.withCacheConfiguration(
"cache1", CouchbaseCacheConfiguration
.defaultCacheConfig().entryExpiry(Duration.ofSeconds(10))
)
.withCacheConfiguration(
"cache2", CouchbaseCacheConfiguration
.defaultCacheConfig().entryExpiry(Duration.ofMinutes(1))
)
}
}
Redis
Properties
spring.cache.cache-names=cache1,cache2
spring.cache.redis.time-to-live=10m
Yaml
spring:
cache:
cache-names: "cache1,cache2"
redis:
time-to-live: "10m"
409
By default, a key prefix is added so that, if two separate caches use the same key,
NOTE Redis does not have overlapping keys and cannot return invalid values. We strongly
recommend keeping this setting enabled if you create your own RedisCacheManager.
Java
import java.time.Duration;
import
org.springframework.boot.autoconfigure.cache.RedisCacheManagerBuilderCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
@Configuration(proxyBeanMethods = false)
public class MyRedisCacheManagerConfiguration {
@Bean
public RedisCacheManagerBuilderCustomizer myRedisCacheManagerBuilderCustomizer() {
return (builder) -> builder
.withCacheConfiguration("cache1", RedisCacheConfiguration
.defaultCacheConfig().entryTtl(Duration.ofSeconds(10)))
.withCacheConfiguration("cache2", RedisCacheConfiguration
.defaultCacheConfig().entryTtl(Duration.ofMinutes(1)));
410
Kotlin
import org.springframework.boot.autoconfigure.cache.RedisCacheManagerBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.data.redis.cache.RedisCacheConfiguration
import java.time.Duration
@Configuration(proxyBeanMethods = false)
class MyRedisCacheManagerConfiguration {
@Bean
fun myRedisCacheManagerBuilderCustomizer(): RedisCacheManagerBuilderCustomizer {
return RedisCacheManagerBuilderCustomizer { builder ->
builder
.withCacheConfiguration(
"cache1", RedisCacheConfiguration
.defaultCacheConfig().entryTtl(Duration.ofSeconds(10))
)
.withCacheConfiguration(
"cache2", RedisCacheConfiguration
.defaultCacheConfig().entryTtl(Duration.ofMinutes(1))
)
}
}
Caffeine
Caffeine is a Java 8 rewrite of Guava’s cache that supersedes support for Guava. If Caffeine is
present, a CaffeineCacheManager (provided by the spring-boot-starter-cache “Starter”) is auto-
configured. Caches can be created on startup by setting the spring.cache.cache-names property and
can be customized by one of the following (in the indicated order):
For instance, the following configuration creates cache1 and cache2 caches with a maximum size of
500 and a time to live of 10 minutes
Properties
spring.cache.cache-names=cache1,cache2
spring.cache.caffeine.spec=maximumSize=500,expireAfterAccess=600s
411
Yaml
spring:
cache:
cache-names: "cache1,cache2"
caffeine:
spec: "maximumSize=500,expireAfterAccess=600s"
Cache2k
Caches can be created on startup by setting the spring.cache.cache-names property. Cache defaults
can be customized using a Cache2kBuilderCustomizer bean. The following example shows a
customizer that configures the capacity of the cache to 200 entries, with an expiration of 5 minutes:
Java
import java.util.concurrent.TimeUnit;
import org.springframework.boot.autoconfigure.cache.Cache2kBuilderCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyCache2kDefaultsConfiguration {
@Bean
public Cache2kBuilderCustomizer myCache2kDefaultsCustomizer() {
return (builder) -> builder.entryCapacity(200)
.expireAfterWrite(5, TimeUnit.MINUTES);
}
412
Kotlin
import org.springframework.boot.autoconfigure.cache.Cache2kBuilderCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import java.util.concurrent.TimeUnit
@Configuration(proxyBeanMethods = false)
class MyCache2kDefaultsConfiguration {
@Bean
fun myCache2kDefaultsCustomizer(): Cache2kBuilderCustomizer {
return Cache2kBuilderCustomizer { builder ->
builder.entryCapacity(200)
.expireAfterWrite(5, TimeUnit.MINUTES)
}
}
}
Simple
If none of the other providers can be found, a simple implementation using a ConcurrentHashMap as
the cache store is configured. This is the default if no caching library is present in your application.
By default, caches are created as needed, but you can restrict the list of available caches by setting
the cache-names property. For instance, if you want only cache1 and cache2 caches, set the cache-
names property as follows:
Properties
spring.cache.cache-names=cache1,cache2
Yaml
spring:
cache:
cache-names: "cache1,cache2"
If you do so and your application uses a cache not listed, then it fails at runtime when the cache is
needed, but not on startup. This is similar to the way the "real" cache providers behave if you use
an undeclared cache.
None
413
Properties
spring.cache.type=none
Yaml
spring:
cache:
type: "none"
11.2. Hazelcast
If Hazelcast is on the classpath and a suitable configuration is found, Spring Boot auto-configures a
HazelcastInstance that you can inject in your application.
Spring Boot first attempts to create a client by checking the following configuration options:
If a client can not be created, Spring Boot attempts to configure an embedded server. If you define a
com.hazelcast.config.Config bean, Spring Boot uses that. If your configuration defines an instance
name, Spring Boot tries to locate an existing instance rather than creating a new one.
You could also specify the Hazelcast configuration file to use through configuration, as shown in the
following example:
Properties
spring.hazelcast.config=classpath:config/my-hazelcast.xml
Yaml
spring:
hazelcast:
config: "classpath:config/my-hazelcast.xml"
Otherwise, Spring Boot tries to find the Hazelcast configuration from the default locations:
hazelcast.xml in the working directory or at the root of the classpath, or a YAML counterpart in the
same locations. We also check if the hazelcast.config system property is set. See the Hazelcast
documentation for more details.
414
By default, @SpringAware on Hazelcast components is supported. The ManagementContext
TIP can be overridden by declaring a HazelcastConfigCustomizer bean with an @Order
higher than zero.
Spring Boot also has explicit caching support for Hazelcast. If caching is enabled,
NOTE
the HazelcastInstance is automatically wrapped in a CacheManager implementation.
Beans of the following types are automatically picked up and associated with the Scheduler:
• JobDetail: defines a particular Job. JobDetail instances can be built with the JobBuilder API.
• Calendar.
Properties
spring.quartz.job-store-type=jdbc
Yaml
spring:
quartz:
job-store-type: "jdbc"
When the JDBC store is used, the schema can be initialized on startup, as shown in the following
example:
Properties
spring.quartz.jdbc.initialize-schema=always
Yaml
spring:
quartz:
jdbc:
initialize-schema: "always"
415
By default, the database is detected and initialized by using the standard
scripts provided with the Quartz library. These scripts drop existing tables,
WARNING
deleting all triggers on every restart. It is also possible to provide a custom
script by setting the spring.quartz.jdbc.schema property.
To have Quartz use a DataSource other than the application’s main DataSource, declare a DataSource
bean, annotating its @Bean method with @QuartzDataSource. Doing so ensures that the Quartz-specific
DataSource is used by both the SchedulerFactoryBean and for schema initialization. Similarly, to have
Quartz use a TransactionManager other than the application’s main TransactionManager declare a
TransactionManager bean, annotating its @Bean method with @QuartzTransactionManager.
By default, jobs created by configuration will not overwrite already registered jobs that have been
read from a persistent job store. To enable overwriting existing job definitions set the
spring.quartz.overwrite-existing-jobs property.
In particular, an Executor bean is not associated with the scheduler as Quartz offers
a way to configure the scheduler through spring.quartz.properties. If you need to
NOTE
customize the task executor, consider implementing
SchedulerFactoryBeanCustomizer.
Jobs can define setters to inject data map properties. Regular beans can also be injected in a similar
manner, as shown in the following example:
416
Java
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.scheduling.quartz.QuartzJobBean;
@Override
protected void executeInternal(JobExecutionContext context) throws
JobExecutionException {
this.myService.someMethod(context.getFireTime(), this.name);
}
417
Kotlin
import org.quartz.JobExecutionContext
import org.springframework.scheduling.quartz.QuartzJobBean
See the reference documentation for a detailed explanation of how you can use
TIP
JavaMailSender.
In particular, certain default timeout values are infinite, and you may want to change that to avoid
having a thread blocked by an unresponsive mail server, as shown in the following example:
Properties
spring.mail.properties[mail.smtp.connectiontimeout]=5000
spring.mail.properties[mail.smtp.timeout]=3000
spring.mail.properties[mail.smtp.writetimeout]=5000
418
Yaml
spring:
mail:
properties:
"[mail.smtp.connectiontimeout]": 5000
"[mail.smtp.timeout]": 3000
"[mail.smtp.writetimeout]": 5000
Properties
spring.mail.jndi-name=mail/Session
Yaml
spring:
mail:
jndi-name: "mail/Session"
When a jndi-name is set, it takes precedence over all other Session-related settings.
11.5. Validation
The method validation feature supported by Bean Validation 1.1 is automatically enabled as long as
a JSR-303 implementation (such as Hibernate validator) is on the classpath. This lets bean methods
be annotated with jakarta.validation constraints on their parameters and/or on their return value.
Target classes with such annotated methods need to be annotated with the @Validated annotation at
the type level for their methods to be searched for inline constraint annotations.
For instance, the following service triggers the validation of the first argument, making sure its size
is between 8 and 10:
419
Java
import jakarta.validation.constraints.Size;
import org.springframework.stereotype.Service;
import org.springframework.validation.annotation.Validated;
@Service
@Validated
public class MyBean {
Kotlin
import jakarta.validation.constraints.Size
import org.springframework.stereotype.Service
import org.springframework.validation.annotation.Validated
@Service
@Validated
class MyBean {
The application’s MessageSource is used when resolving {parameters} in constraint messages. This
allows you to use your application’s messages.properties files for Bean Validation messages. Once
the parameters have been resolved, message interpolation is completed using Bean Validation’s
default interpolator.
420
11.6.1. RestTemplate
If you need to call remote REST services from your application, you can use the Spring Framework’s
RestTemplate class. Since RestTemplate instances often need to be customized before being used,
Spring Boot does not provide any single auto-configured RestTemplate bean. It does, however, auto-
configure a RestTemplateBuilder, which can be used to create RestTemplate instances when needed.
The auto-configured RestTemplateBuilder ensures that sensible HttpMessageConverters are applied to
RestTemplate instances.
Java
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
@Service
public class MyService {
421
Kotlin
import org.springframework.boot.web.client.RestTemplateBuilder
import org.springframework.stereotype.Service
import org.springframework.web.client.RestTemplate
@Service
class MyService(restTemplateBuilder: RestTemplateBuilder) {
init {
restTemplate = restTemplateBuilder.build()
}
RestTemplateBuilder includes a number of useful methods that can be used to quickly configure a
RestTemplate. For example, to add BASIC authentication support, you can use
builder.basicAuthentication("user", "password").build().
RestTemplate Customization
There are three main approaches to RestTemplate customization, depending on how broadly you
want the customizations to apply.
To make the scope of any customizations as narrow as possible, inject the auto-configured
RestTemplateBuilder and then call its methods as required. Each method call returns a new
RestTemplateBuilder instance, so the customizations only affect this use of the builder.
The following example shows a customizer that configures the use of a proxy for all hosts except
192.168.0.5:
422
Java
import org.apache.hc.client5.http.classic.HttpClient;
import org.apache.hc.client5.http.impl.classic.HttpClientBuilder;
import org.apache.hc.client5.http.impl.routing.DefaultProxyRoutePlanner;
import org.apache.hc.client5.http.routing.HttpRoutePlanner;
import org.apache.hc.core5.http.HttpException;
import org.apache.hc.core5.http.HttpHost;
import org.apache.hc.core5.http.protocol.HttpContext;
import org.springframework.boot.web.client.RestTemplateCustomizer;
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.web.client.RestTemplate;
@Override
public void customize(RestTemplate restTemplate) {
HttpRoutePlanner routePlanner = new CustomRoutePlanner(new
HttpHost("proxy.example.com"));
HttpClient httpClient =
HttpClientBuilder.create().setRoutePlanner(routePlanner).build();
restTemplate.setRequestFactory(new
HttpComponentsClientHttpRequestFactory(httpClient));
}
CustomRoutePlanner(HttpHost proxy) {
super(proxy);
}
@Override
protected HttpHost determineProxy(HttpHost target, HttpContext context) throws
HttpException {
if (target.getHostName().equals("192.168.0.5")) {
return null;
}
return super.determineProxy(target, context);
}
423
Kotlin
import org.apache.hc.client5.http.classic.HttpClient
import org.apache.hc.client5.http.impl.classic.HttpClientBuilder
import org.apache.hc.client5.http.impl.routing.DefaultProxyRoutePlanner
import org.apache.hc.client5.http.routing.HttpRoutePlanner
import org.apache.hc.core5.http.HttpException
import org.apache.hc.core5.http.HttpHost
import org.apache.hc.core5.http.protocol.HttpContext
import org.springframework.boot.web.client.RestTemplateCustomizer
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory
import org.springframework.web.client.RestTemplate
@Throws(HttpException::class)
public override fun determineProxy(target: HttpHost, context: HttpContext):
HttpHost? {
if (target.hostName == "192.168.0.5") {
return null
}
return super.determineProxy(target, context)
}
Finally, you can define your own RestTemplateBuilder bean. Doing so will replace the auto-
configured builder. If you want any RestTemplateCustomizer beans to be applied to your custom
builder, as the auto-configuration would have done, configure it using a
RestTemplateBuilderConfigurer. The following example exposes a RestTemplateBuilder that matches
what Spring Boot’s auto-configuration would have done, except that custom connect and read
timeouts are also specified:
424
Java
import java.time.Duration;
import
org.springframework.boot.autoconfigure.web.client.RestTemplateBuilderConfigurer;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyRestTemplateBuilderConfiguration {
@Bean
public RestTemplateBuilder restTemplateBuilder(RestTemplateBuilderConfigurer
configurer) {
return configurer.configure(new RestTemplateBuilder())
.setConnectTimeout(Duration.ofSeconds(5))
.setReadTimeout(Duration.ofSeconds(2));
}
Kotlin
import org.springframework.boot.autoconfigure.web.client.RestTemplateBuilderConfigurer
import org.springframework.boot.web.client.RestTemplateBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import java.time.Duration
@Configuration(proxyBeanMethods = false)
class MyRestTemplateBuilderConfiguration {
@Bean
fun restTemplateBuilder(configurer: RestTemplateBuilderConfigurer):
RestTemplateBuilder {
return
configurer.configure(RestTemplateBuilder()).setConnectTimeout(Duration.ofSeconds(5))
.setReadTimeout(Duration.ofSeconds(2))
}
The most extreme (and rarely used) option is to create your own RestTemplateBuilder bean without
using a configurer. In addition to replacing the auto-configured builder, this also prevents any
RestTemplateCustomizer beans from being used.
425
RestTemplate SSL Support
If you need custom SSL configuration on the RestTemplate, you can apply an SSL bundle to the
RestTemplateBuilder as shown in this example:
Java
import org.springframework.boot.docs.io.restclient.resttemplate.Details;
import org.springframework.boot.ssl.SslBundles;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
@Service
public class MyService {
426
Kotlin
import org.springframework.boot.docs.io.restclient.resttemplate.Details
import org.springframework.boot.ssl.SslBundles
import org.springframework.boot.web.client.RestTemplateBuilder
import org.springframework.stereotype.Service
import org.springframework.web.client.RestTemplate
@Service
class MyService(restTemplateBuilder: RestTemplateBuilder, sslBundles: SslBundles) {
init {
restTemplate =
restTemplateBuilder.setSslBundle(sslBundles.getBundle("mybundle")).build()
}
11.6.2. WebClient
If you have Spring WebFlux on your classpath, you can also choose to use WebClient to call remote
REST services. Compared to RestTemplate, this client has a more functional feel and is fully reactive.
You can learn more about the WebClient in the dedicated section in the Spring Framework docs.
Spring Boot creates and pre-configures a WebClient.Builder for you. It is strongly advised to inject it
in your components and use it to create WebClient instances. Spring Boot is configuring that builder
to share HTTP resources, reflect codecs setup in the same fashion as the server ones (see WebFlux
HTTP codecs auto-configuration), and more.
427
Java
import reactor.core.publisher.Mono;
import org.springframework.stereotype.Service;
import org.springframework.web.reactive.function.client.WebClient;
@Service
public class MyService {
Kotlin
import org.springframework.stereotype.Service
import org.springframework.web.reactive.function.client.WebClient
import reactor.core.publisher.Mono
@Service
class MyService(webClientBuilder: WebClient.Builder) {
init {
webClient = webClientBuilder.baseUrl("https://example.org").build()
}
WebClient Runtime
Spring Boot will auto-detect which ClientHttpConnector to use to drive WebClient, depending on the
libraries available on the application classpath. For now, Reactor Netty, Jetty ReactiveStream client,
Apache HttpClient, and the JDK’s HttpClient are supported.
428
The spring-boot-starter-webflux starter depends on io.projectreactor.netty:reactor-netty by
default, which brings both server and client implementations. If you choose to use Jetty as a
reactive server instead, you should add a dependency on the Jetty Reactive HTTP client library,
org.eclipse.jetty:jetty-reactive-httpclient. Using the same technology for server and client has
its advantages, as it will automatically share HTTP resources between client and server.
Developers can override the resource configuration for Jetty and Reactor Netty by providing a
custom ReactorResourceFactory or JettyResourceFactory bean - this will be applied to both clients
and servers.
If you wish to override that choice for the client, you can define your own ClientHttpConnector bean
and have full control over the client configuration.
You can learn more about the WebClient configuration options in the Spring Framework reference
documentation.
WebClient Customization
There are three main approaches to WebClient customization, depending on how broadly you want
the customizations to apply.
To make the scope of any customizations as narrow as possible, inject the auto-configured
WebClient.Builder and then call its methods as required. WebClient.Builder instances are stateful:
Any change on the builder is reflected in all clients subsequently created with it. If you want to
create several clients with the same builder, you can also consider cloning the builder with
WebClient.Builder other = builder.clone();.
Finally, you can fall back to the original API and use WebClient.create(). In that case, no auto-
configuration or WebClientCustomizer is applied.
If you need custom SSL configuration on the ClientHttpConnector used by the WebClient, you can
inject a WebClientSsl instance that can be used with the builder’s apply method.
The WebClientSsl interface provides access to any SSL bundles that you have defined in your
application.properties or application.yaml file.
429
Java
import reactor.core.publisher.Mono;
import
org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientSsl;
import org.springframework.boot.docs.io.restclient.webclient.Details;
import org.springframework.stereotype.Service;
import org.springframework.web.reactive.function.client.WebClient;
@Service
public class MyService {
430
Kotlin
import
org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientSsl
import org.springframework.boot.docs.io.restclient.webclient.Details
import org.springframework.stereotype.Service
import org.springframework.web.reactive.function.client.WebClient
import reactor.core.publisher.Mono
@Service
class MyService(webClientBuilder: WebClient.Builder, ssl: WebClientSsl) {
init {
webClient =
webClientBuilder.baseUrl("https://example.org").apply(ssl.fromBundle("mybundle")).buil
d()
}
The Spring Web Services features can be easily accessed with the spring-boot-starter-webservices
module.
SimpleWsdl11Definition and SimpleXsdSchema beans can be automatically created for your WSDLs
and XSDs respectively. To do so, configure their location, as shown in the following example:
Properties
spring.webservices.wsdl-locations=classpath:/wsdl
Yaml
spring:
webservices:
wsdl-locations: "classpath:/wsdl"
431
11.7.1. Calling Web Services with WebServiceTemplate
If you need to call remote Web services from your application, you can use the WebServiceTemplate
class. Since WebServiceTemplate instances often need to be customized before being used, Spring
Boot does not provide any single auto-configured WebServiceTemplate bean. It does, however, auto-
configure a WebServiceTemplateBuilder, which can be used to create WebServiceTemplate instances
when needed.
Java
import org.springframework.boot.webservices.client.WebServiceTemplateBuilder;
import org.springframework.stereotype.Service;
import org.springframework.ws.client.core.WebServiceTemplate;
import org.springframework.ws.soap.client.core.SoapActionCallback;
@Service
public class MyService {
432
Kotlin
import org.springframework.boot.webservices.client.WebServiceTemplateBuilder
import org.springframework.stereotype.Service
import org.springframework.ws.client.core.WebServiceTemplate
import org.springframework.ws.soap.client.core.SoapActionCallback
@Service
class MyService(webServiceTemplateBuilder: WebServiceTemplateBuilder) {
init {
webServiceTemplate = webServiceTemplateBuilder.build()
}
433
Java
import java.time.Duration;
import org.springframework.boot.webservices.client.HttpWebServiceMessageSenderBuilder;
import org.springframework.boot.webservices.client.WebServiceTemplateBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.ws.client.core.WebServiceTemplate;
import org.springframework.ws.transport.WebServiceMessageSender;
@Configuration(proxyBeanMethods = false)
public class MyWebServiceTemplateConfiguration {
@Bean
public WebServiceTemplate webServiceTemplate(WebServiceTemplateBuilder builder) {
WebServiceMessageSender sender = new HttpWebServiceMessageSenderBuilder()
.setConnectTimeout(Duration.ofSeconds(5))
.setReadTimeout(Duration.ofSeconds(2))
.build();
return builder.messageSenders(sender).build();
}
Kotlin
import org.springframework.boot.webservices.client.HttpWebServiceMessageSenderBuilder
import org.springframework.boot.webservices.client.WebServiceTemplateBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.ws.client.core.WebServiceTemplate
import java.time.Duration
@Configuration(proxyBeanMethods = false)
class MyWebServiceTemplateConfiguration {
@Bean
fun webServiceTemplate(builder: WebServiceTemplateBuilder): WebServiceTemplate {
val sender = HttpWebServiceMessageSenderBuilder()
.setConnectTimeout(Duration.ofSeconds(5))
.setReadTimeout(Duration.ofSeconds(2))
.build()
return builder.messageSenders(sender).build()
}
434
11.8. Distributed Transactions With JTA
Spring Boot supports distributed JTA transactions across multiple XA resources by using a
transaction manager retrieved from JNDI.
If you package your Spring Boot application as a war or ear file and deploy it to a Jakarta EE
application server, you can use your application server’s built-in transaction manager. Spring Boot
tries to auto-configure a transaction manager by looking at common JNDI locations
(java:comp/UserTransaction, java:comp/TransactionManager, and so on). When using a transaction
service provided by your application server, you generally also want to ensure that all resources
are managed by the server and exposed over JNDI. Spring Boot tries to auto-configure JMS by
looking for a ConnectionFactory at the JNDI path (java:/JmsXA or java:/XAConnectionFactory), and
you can use the spring.datasource.jndi-name property to configure your DataSource.
When using JTA, the primary JMS ConnectionFactory bean is XA-aware and participates in
distributed transactions. You can inject into your bean without needing to use any @Qualifier:
Java
Kotlin
In some situations, you might want to process certain JMS messages by using a non-XA
ConnectionFactory. For example, your JMS processing logic might take longer than the XA timeout.
If you want to use a non-XA ConnectionFactory, you can the nonXaJmsConnectionFactory bean:
Java
435
Kotlin
For consistency, the jmsConnectionFactory bean is also provided by using the bean alias
xaJmsConnectionFactory:
Java
Kotlin
The next few sections go into detail about deploying applications to cloud platforms. You can read
about building container images in the next section or skip to the production-ready features
section.
436
Chapter 12. Container Images
Spring Boot applications can be containerized using Dockerfiles, or by using Cloud Native
Buildpacks to create optimized docker compatible container images that you can run anywhere.
To make it easier to create optimized Docker images, Spring Boot supports adding a layer index file
to the jar. It provides a list of layers and the parts of the jar that should be contained within them.
The list of layers in the index is ordered based on the order in which the layers should be added to
the Docker/OCI image. Out-of-the-box, the following layers are supported:
- "dependencies":
- BOOT-INF/lib/library1.jar
- BOOT-INF/lib/library2.jar
- "spring-boot-loader":
- org/springframework/boot/loader/JarLauncher.class
- org/springframework/boot/loader/jar/JarEntry.class
- "snapshot-dependencies":
- BOOT-INF/lib/library3-SNAPSHOT.jar
- "application":
- META-INF/MANIFEST.MF
- BOOT-INF/classes/a/b/C.class
This layering is designed to separate code based on how likely it is to change between application
builds. Library code is less likely to change between builds, so it is placed in its own layers to allow
tooling to re-use the layers from cache. Application code is more likely to change between builds so
it is isolated in a separate layer.
437
Spring Boot also supports layering for war files with the help of a layers.idx.
For Maven, see the packaging layered jar or war section for more details on adding a layer index to
the archive. For Gradle, see the packaging layered jar or war section of the Gradle plugin
documentation.
12.2. Dockerfiles
While it is possible to convert a Spring Boot fat jar into a docker image with just a few lines in the
Dockerfile, we will use the layering feature to create an optimized docker image. When you create a
jar containing the layers index file, the spring-boot-jarmode-layertools jar will be added as a
dependency to your jar. With this jar on the classpath, you can launch your application in a special
mode which allows the bootstrap code to run something entirely different from your application,
for example, something that extracts the layers.
The layertools mode can not be used with a fully executable Spring Boot
CAUTION archive that includes a launch script. Disable launch script configuration when
building a jar file that is intended to be used with layertools.
Here’s how you can launch your jar with a layertools jar mode:
Usage:
java -Djarmode=layertools -jar my-app.jar
Available commands:
list List layers from the jar that can be extracted
extract Extracts layers from the jar for image creation
help Help about any command
The extract command can be used to easily split the application into layers to be added to the
dockerfile. Here is an example of a Dockerfile using jarmode.
438
FROM eclipse-temurin:17-jre as builder
WORKDIR application
ARG JAR_FILE=target/*.jar
COPY ${JAR_FILE} application.jar
RUN java -Djarmode=layertools -jar application.jar extract
FROM eclipse-temurin:17-jre
WORKDIR application
COPY --from=builder application/dependencies/ ./
COPY --from=builder application/spring-boot-loader/ ./
COPY --from=builder application/snapshot-dependencies/ ./
COPY --from=builder application/application/ ./
ENTRYPOINT ["java", "org.springframework.boot.loader.JarLauncher"]
Assuming the above Dockerfile is in the current directory, your docker image can be built with
docker build ., or optionally specifying the path to your application jar, as shown in the following
example:
This is a multi-stage dockerfile. The builder stage extracts the directories that are needed later. Each
of the COPY commands relates to the layers extracted by the jarmode.
Of course, a Dockerfile can be written without using the jarmode. You can use some combination of
unzip and mv to move things to the right layer but jarmode simplifies that.
With Cloud Native Buildpacks, you can create Docker compatible images that you can run
anywhere. Spring Boot includes buildpack support directly for both Maven and Gradle. This means
you can just type a single command and quickly get a sensible image into your locally running
Docker daemon.
See the individual plugin documentation on how to use buildpacks with Maven and Gradle.
The Paketo Spring Boot buildpack has also been updated to support the layers.idx
NOTE file so any customization that is applied to it will be reflected in the image created
by the buildpack.
439
In order to achieve reproducible builds and container image caching, Buildpacks
can manipulate the application resources metadata (such as the file "last modified"
information). You should ensure that your application does not rely on that
NOTE
metadata at runtime. Spring Boot can use that information when serving static
resources, but this can be disabled with spring.web.resources.cache.use-last-
modified.
440
Chapter 13. Production-ready Features
Spring Boot includes a number of additional features to help you monitor and manage your
application when you push it to production. You can choose to manage and monitor your
application by using HTTP endpoints or with JMX. Auditing, health, and metrics gathering can also
be automatically applied to your application.
Definition of Actuator
To add the actuator to a Maven-based project, add the following “Starter” dependency:
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-actuator'
}
13.2. Endpoints
Actuator endpoints let you monitor and interact with your application. Spring Boot includes a
number of built-in endpoints and lets you add your own. For example, the health endpoint provides
basic application health information.
You can enable or disable each individual endpoint and expose them (make them remotely
accessible) over HTTP or JMX. An endpoint is considered to be available when it is both enabled
and exposed. The built-in endpoints are auto-configured only when they are available. Most
applications choose exposure over HTTP, where the ID of the endpoint and a prefix of /actuator is
mapped to a URL. For example, by default, the health endpoint is mapped to /actuator/health.
441
To learn more about the Actuator’s endpoints and their request and response formats,
TIP
see the separate API documentation (HTML or PDF).
ID Description
auditevents Exposes audit events information for the current application.
Requires an AuditEventRepository bean.
beans Displays a complete list of all the Spring beans in your application.
caches Exposes available caches.
conditions Shows the conditions that were evaluated on configuration and auto-
configuration classes and the reasons why they did or did not match.
configprops Displays a collated list of all @ConfigurationProperties.
env Exposes properties from Spring’s ConfigurableEnvironment.
flyway Shows any Flyway database migrations that have been applied.
Requires one or more Flyway beans.
health Shows application health information.
httpexchanges Displays HTTP exchange information (by default, the last 100 HTTP
request-response exchanges). Requires an HttpExchangeRepository
bean.
info Displays arbitrary application info.
integrationgraph Shows the Spring Integration graph. Requires a dependency on
spring-integration-core.
loggers Shows and modifies the configuration of loggers in the application.
liquibase Shows any Liquibase database migrations that have been applied.
Requires one or more Liquibase beans.
metrics Shows “metrics” information for the current application.
mappings Displays a collated list of all @RequestMapping paths.
quartz Shows information about Quartz Scheduler jobs.
scheduledtasks Displays the scheduled tasks in your application.
sessions Allows retrieval and deletion of user sessions from a Spring Session-
backed session store. Requires a servlet-based web application that
uses Spring Session.
shutdown Lets the application be gracefully shutdown. Only works when using
jar packaging. Disabled by default.
startup Shows the startup steps data collected by the ApplicationStartup.
Requires the SpringApplication to be configured with a
BufferingApplicationStartup.
threaddump Performs a thread dump.
442
If your application is a web application (Spring MVC, Spring WebFlux, or Jersey), you can use the
following additional endpoints:
ID Description
heapdump Returns a heap dump file. On a HotSpot JVM, an HPROF-format file is
returned. On an OpenJ9 JVM, a PHD-format file is returned.
logfile Returns the contents of the logfile (if the logging.file.name or the
logging.file.path property has been set). Supports the use of the
HTTP Range header to retrieve part of the log file’s content.
prometheus Exposes metrics in a format that can be scraped by a Prometheus
server. Requires a dependency on micrometer-registry-prometheus.
By default, all endpoints except for shutdown are enabled. To configure the enablement of an
endpoint, use its management.endpoint.<id>.enabled property. The following example enables the
shutdown endpoint:
Properties
management.endpoint.shutdown.enabled=true
Yaml
management:
endpoint:
shutdown:
enabled: true
If you prefer endpoint enablement to be opt-in rather than opt-out, set the
management.endpoints.enabled-by-default property to false and use individual endpoint enabled
properties to opt back in. The following example enables the info endpoint and disables all other
endpoints:
Properties
management.endpoints.enabled-by-default=false
management.endpoint.info.enabled=true
443
Yaml
management:
endpoints:
enabled-by-default: false
endpoint:
info:
enabled: true
Disabled endpoints are removed entirely from the application context. If you want
NOTE to change only the technologies over which an endpoint is exposed, use the include
and exclude properties instead.
By default, only the health endpoint is exposed over HTTP and JMX. Since Endpoints may contain
sensitive information, you should carefully consider when to expose them.
To change which endpoints are exposed, use the following technology-specific include and exclude
properties:
Property Default
management.endpoints.jmx.exposure.exclude
management.endpoints.jmx.exposure.include health
management.endpoints.web.exposure.exclude
management.endpoints.web.exposure.include health
The include property lists the IDs of the endpoints that are exposed. The exclude property lists the
IDs of the endpoints that should not be exposed. The exclude property takes precedence over the
include property. You can configure both the include and the exclude properties with a list of
endpoint IDs.
For example, to only expose the health and info endpoints over JMX, use the following property:
Properties
management.endpoints.jmx.exposure.include=health,info
Yaml
management:
endpoints:
jmx:
exposure:
include: "health,info"
* can be used to select all endpoints. For example, to expose everything over HTTP except the env
444
and beans endpoints, use the following properties:
Properties
management.endpoints.web.exposure.include=*
management.endpoints.web.exposure.exclude=env,beans
Yaml
management:
endpoints:
web:
exposure:
include: "*"
exclude: "env,beans"
* has a special meaning in YAML, so be sure to add quotation marks if you want to
NOTE
include (or exclude) all endpoints.
If your application is exposed publicly, we strongly recommend that you also secure
NOTE
your endpoints.
If you want to implement your own strategy for when endpoints are exposed, you can
TIP
register an EndpointFilter bean.
13.2.3. Security
For security purposes, only the /health endpoint is exposed over HTTP by default. You can use the
management.endpoints.web.exposure.include property to configure the endpoints that are exposed.
If Spring Security is on the classpath and no other SecurityFilterChain bean is present, all actuators
other than /health are secured by Spring Boot auto-configuration. If you define a custom
SecurityFilterChain bean, Spring Boot auto-configuration backs off and lets you fully control the
actuator access rules.
If you wish to configure custom security for HTTP endpoints (for example, to allow only users with
a certain role to access them), Spring Boot provides some convenient RequestMatcher objects that
you can use in combination with Spring Security.
A typical Spring Security configuration might look something like the following example:
445
Java
import
org.springframework.boot.actuate.autoconfigure.security.servlet.EndpointRequest;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MySecurityConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.securityMatcher(EndpointRequest.toAnyEndpoint());
http.authorizeHttpRequests((requests) ->
requests.anyRequest().hasRole("ENDPOINT_ADMIN"));
http.httpBasic(withDefaults());
return http.build();
}
Kotlin
import org.springframework.boot.actuate.autoconfigure.security.servlet.EndpointRequest
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.security.config.Customizer.withDefaults
import org.springframework.security.config.annotation.web.builders.HttpSecurity
import org.springframework.security.web.SecurityFilterChain
@Configuration(proxyBeanMethods = false)
class MySecurityConfiguration {
@Bean
fun securityFilterChain(http: HttpSecurity): SecurityFilterChain {
http.securityMatcher(EndpointRequest.toAnyEndpoint()).authorizeHttpRequests {
requests ->
requests.anyRequest().hasRole("ENDPOINT_ADMIN")
}
http.httpBasic(withDefaults())
return http.build()
}
446
The preceding example uses EndpointRequest.toAnyEndpoint() to match a request to any endpoint
and then ensures that all have the ENDPOINT_ADMIN role. Several other matcher methods are also
available on EndpointRequest. See the API documentation (HTML or PDF) for details.
If you deploy applications behind a firewall, you may prefer that all your actuator endpoints can be
accessed without requiring authentication. You can do so by changing the
management.endpoints.web.exposure.include property, as follows:
Properties
management.endpoints.web.exposure.include=*
Yaml
management:
endpoints:
web:
exposure:
include: "*"
Additionally, if Spring Security is present, you would need to add custom security configuration that
allows unauthenticated access to the endpoints, as the following example shows:
Java
import
org.springframework.boot.actuate.autoconfigure.security.servlet.EndpointRequest;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MySecurityConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.securityMatcher(EndpointRequest.toAnyEndpoint());
http.authorizeHttpRequests((requests) -> requests.anyRequest().permitAll());
return http.build();
}
447
Kotlin
import org.springframework.boot.actuate.autoconfigure.security.servlet.EndpointRequest
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.security.config.annotation.web.builders.HttpSecurity
import org.springframework.security.web.SecurityFilterChain
@Configuration(proxyBeanMethods = false)
class MySecurityConfiguration {
@Bean
fun securityFilterChain(http: HttpSecurity): SecurityFilterChain {
http.securityMatcher(EndpointRequest.toAnyEndpoint()).authorizeHttpRequests {
requests -> requests.anyRequest().permitAll() }
return http.build()
}
In both of the preceding examples, the configuration applies only to the actuator
endpoints. Since Spring Boot’s security configuration backs off completely in the
NOTE
presence of any SecurityFilterChain bean, you need to configure an additional
SecurityFilterChain bean with rules that apply to the rest of the application.
Since Spring Boot relies on Spring Security’s defaults, CSRF protection is turned on by default. This
means that the actuator endpoints that require a POST (shutdown and loggers endpoints), a PUT, or a
DELETE get a 403 (forbidden) error when the default security configuration is in use.
You can find additional information about CSRF protection in the Spring Security Reference Guide.
Endpoints automatically cache responses to read operations that do not take any parameters. To
configure the amount of time for which an endpoint caches a response, use its cache.time-to-live
property. The following example sets the time-to-live of the beans endpoint’s cache to 10 seconds:
Properties
management.endpoint.beans.cache.time-to-live=10s
448
Yaml
management:
endpoint:
beans:
cache:
time-to-live: "10s"
A “discovery page” is added with links to all the endpoints. The “discovery page” is available on
/actuator by default.
To disable the “discovery page”, add the following property to your application properties:
Properties
management.endpoints.web.discovery.enabled=false
Yaml
management:
endpoints:
web:
discovery:
enabled: false
When a custom management context path is configured, the “discovery page” automatically moves
from /actuator to the root of the management context. For example, if the management context
path is /management, the discovery page is available from /management. When the management
context path is set to /, the discovery page is disabled to prevent the possibility of a clash with other
mappings.
Cross-origin resource sharing (CORS) is a W3C specification that lets you specify in a flexible way
what kind of cross-domain requests are authorized. If you use Spring MVC or Spring WebFlux, you
can configure Actuator’s web endpoints to support such scenarios.
CORS support is disabled by default and is only enabled once you have set the
management.endpoints.web.cors.allowed-origins property. The following configuration permits GET
and POST calls from the example.com domain:
449
Properties
management.endpoints.web.cors.allowed-origins=https://example.com
management.endpoints.web.cors.allowed-methods=GET,POST
Yaml
management:
endpoints:
web:
cors:
allowed-origins: "https://example.com"
allowed-methods: "GET,POST"
If you add a @Bean annotated with @Endpoint, any methods annotated with @ReadOperation,
@WriteOperation, or @DeleteOperation are automatically exposed over JMX and, in a web application,
over HTTP as well. Endpoints can be exposed over HTTP by using Jersey, Spring MVC, or Spring
WebFlux. If both Jersey and Spring MVC are available, Spring MVC is used.
The following example exposes a read operation that returns a custom object:
Java
@ReadOperation
public CustomData getData() {
return new CustomData("test", 5);
}
Kotlin
@ReadOperation
fun getData(): CustomData {
return CustomData("test", 5)
}
You can also write technology-specific endpoints by using @JmxEndpoint or @WebEndpoint. These
endpoints are restricted to their respective technologies. For example, @WebEndpoint is exposed only
over HTTP and not over JMX.
Finally, if you need access to web-framework-specific functionality, you can implement servlet or
450
Spring @Controller and @RestController endpoints at the cost of them not being available over JMX
or when using a different web framework.
Receiving Input
Operations on an endpoint receive input through their parameters. When exposed over the web,
the values for these parameters are taken from the URL’s query parameters and from the JSON
request body. When exposed over JMX, the parameters are mapped to the parameters of the
MBean’s operations. Parameters are required by default. They can be made optional by annotating
them with either @javax.annotation.Nullable or @org.springframework.lang.Nullable.
You can map each root property in the JSON request body to a parameter of the endpoint. Consider
the following JSON request body:
{
"name": "test",
"counter": 42
}
You can use this to invoke a write operation that takes String name and int counter parameters, as
the following example shows:
Java
@WriteOperation
public void updateData(String name, int counter) {
// injects "test" and 42
}
Kotlin
@WriteOperation
fun updateData(name: String?, counter: Int) {
// injects "test" and 42
}
Because endpoints are technology agnostic, only simple types can be specified in the
TIP method signature. In particular, declaring a single parameter with a CustomData type
that defines a name and counter properties is not supported.
To let the input be mapped to the operation method’s parameters, Java code that
implements an endpoint should be compiled with -parameters, and Kotlin code that
NOTE implements an endpoint should be compiled with -java-parameters. This will
happen automatically if you use Spring Boot’s Gradle plugin or if you use Maven
and spring-boot-starter-parent.
451
Input Type Conversion
The parameters passed to endpoint operation methods are, if necessary, automatically converted to
the required type. Before calling an operation method, the input received over JMX or HTTP is
converted to the required types by using an instance of ApplicationConversionService as well as any
Converter or GenericConverter beans qualified with @EndpointConverter.
Path
The path of the predicate is determined by the ID of the endpoint and the base path of the web-
exposed endpoints. The default base path is /actuator. For example, an endpoint with an ID of
sessions uses /actuator/sessions as its path in the predicate.
You can further customize the path by annotating one or more parameters of the operation method
with @Selector. Such a parameter is added to the path predicate as a path variable. The variable’s
value is passed into the operation method when the endpoint operation is invoked. If you want to
capture all remaining path elements, you can add @Selector(Match=ALL_REMAINING) to the last
parameter and make it a type that is conversion-compatible with a String[].
HTTP method
The HTTP method of the predicate is determined by the operation type, as shown in the following
table:
Consumes
For a @WriteOperation (HTTP POST) that uses the request body, the consumes clause of the predicate is
application/vnd.spring-boot.actuator.v2+json, application/json. For all other operations, the
consumes clause is empty.
Produces
The produces clause of the predicate can be determined by the produces attribute of the
@DeleteOperation, @ReadOperation, and @WriteOperation annotations. The attribute is optional. If it is
not used, the produces clause is determined automatically.
452
If the operation method returns void or Void, the produces clause is empty. If the operation method
returns a org.springframework.core.io.Resource, the produces clause is application/octet-stream.
For all other operations, the produces clause is application/vnd.spring-boot.actuator.v2+json,
application/json.
The default response status for an endpoint operation depends on the operation type (read, write,
or delete) and what, if anything, the operation returns.
If a @ReadOperation returns a value, the response status will be 200 (OK). If it does not return a
value, the response status will be 404 (Not Found).
If a @WriteOperation or @DeleteOperation returns a value, the response status will be 200 (OK). If it
does not return a value, the response status will be 204 (No Content).
You can use an HTTP range request to request part of an HTTP resource. When using Spring MVC or
Spring Web Flux, operations that return a org.springframework.core.io.Resource automatically
support range requests.
An operation on a web endpoint or a web-specific endpoint extension can receive the current
java.security.Principal or org.springframework.boot.actuate.endpoint.SecurityContext as a
method parameter. The former is typically used in conjunction with @Nullable to provide different
behavior for authenticated and unauthenticated users. The latter is typically used to perform
authorization checks by using its isUserInRole(String) method.
Servlet Endpoints
Controller Endpoints
453
with Spring’s web frameworks but at the expense of portability. The @Endpoint and @WebEndpoint
annotations should be preferred whenever possible.
You can use health information to check the status of your running application. It is often used by
monitoring software to alert someone when a production system goes down. The information
exposed by the health endpoint depends on the management.endpoint.health.show-details and
management.endpoint.health.show-components properties, which can be configured with one of the
following values:
Name Description
never Details are never shown.
when-authorized Details are shown only to authorized users. Authorized roles can be
configured by using management.endpoint.health.roles.
always Details are shown to all users.
The default value is never. A user is considered to be authorized when they are in one or more of
the endpoint’s roles. If the endpoint has no configured roles (the default), all authenticated users
are considered to be authorized. You can configure the roles by using the
management.endpoint.health.roles property.
If you have secured your application and wish to use always, your security
NOTE configuration must permit access to the health endpoint for both authenticated and
unauthenticated users.
Health information is collected from the content of a HealthContributorRegistry (by default, all
HealthContributor instances defined in your ApplicationContext). Spring Boot includes a number of
auto-configured HealthContributors, and you can also write your own.
By default, the final system health is derived by a StatusAggregator, which sorts the statuses from
each HealthIndicator based on an ordered list of statuses. The first status in the sorted list is used as
the overall health status. If no HealthIndicator returns a status that is known to the
StatusAggregator, an UNKNOWN status is used.
You can use the HealthContributorRegistry to register and unregister health indicators
TIP
at runtime.
Auto-configured HealthIndicators
When appropriate, Spring Boot auto-configures the HealthIndicators listed in the following table.
You can also enable or disable selected indicators by configuring management.health.key.enabled,
454
with the key listed in the following table:
TIP You can disable them all by setting the management.health.defaults.enabled property.
To provide custom health information, you can register Spring beans that implement the
HealthIndicator interface. You need to provide an implementation of the health() method and
return a Health response. The Health response should include a status and can optionally include
additional details to be displayed. The following code shows a sample HealthIndicator
implementation:
455
Java
import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.HealthIndicator;
import org.springframework.stereotype.Component;
@Component
public class MyHealthIndicator implements HealthIndicator {
@Override
public Health health() {
int errorCode = check();
if (errorCode != 0) {
return Health.down().withDetail("Error Code", errorCode).build();
}
return Health.up().build();
}
Kotlin
import org.springframework.boot.actuate.health.Health
import org.springframework.boot.actuate.health.HealthIndicator
import org.springframework.stereotype.Component
@Component
class MyHealthIndicator : HealthIndicator {
456
The identifier for a given HealthIndicator is the name of the bean without the
NOTE HealthIndicator suffix, if it exists. In the preceding example, the health information
is available in an entry named my.
Health indicators are usually called over HTTP and need to respond before any
connection timeouts. Spring Boot will log a warning message for any health indicator
TIP that takes longer than 10 seconds to respond. If you want to configure this threshold,
you can use the management.endpoint.health.logging.slow-indicator-threshold
property.
In addition to Spring Boot’s predefined Status types, Health can return a custom Status that
represents a new system state. In such cases, you also need to provide a custom implementation of
the StatusAggregator interface, or you must configure the default implementation by using the
management.endpoint.health.status.order configuration property.
For example, assume a new Status with a code of FATAL is being used in one of your HealthIndicator
implementations. To configure the severity order, add the following property to your application
properties:
Properties
management.endpoint.health.status.order=fatal,down,out-of-service,unknown,up
Yaml
management:
endpoint:
health:
status:
order: "fatal,down,out-of-service,unknown,up"
The HTTP status code in the response reflects the overall health status. By default, OUT_OF_SERVICE
and DOWN map to 503. Any unmapped health statuses, including UP, map to 200. You might also want
to register custom status mappings if you access the health endpoint over HTTP. Configuring a
custom mapping disables the defaults mappings for DOWN and OUT_OF_SERVICE. If you want to retain
the default mappings, you must explicitly configure them, alongside any custom mappings. For
example, the following property maps FATAL to 503 (service unavailable) and retains the default
mappings for DOWN and OUT_OF_SERVICE:
Properties
management.endpoint.health.status.http-mapping.down=503
management.endpoint.health.status.http-mapping.fatal=503
management.endpoint.health.status.http-mapping.out-of-service=503
457
Yaml
management:
endpoint:
health:
status:
http-mapping:
down: 503
fatal: 503
out-of-service: 503
TIP If you need more control, you can define your own HttpCodeStatusMapper bean.
The following table shows the default status mappings for the built-in statuses:
Status Mapping
DOWN SERVICE_UNAVAILABLE (503)
OUT_OF_SERVICE SERVICE_UNAVAILABLE (503)
UP No mapping by default, so HTTP status is 200
UNKNOWN No mapping by default, so HTTP status is 200
For reactive applications, such as those that use Spring WebFlux, ReactiveHealthContributor
provides a non-blocking contract for getting application health. Similar to a traditional
HealthContributor, health information is collected from the content of a
ReactiveHealthContributorRegistry (by default, all HealthContributor and ReactiveHealthContributor
instances defined in your ApplicationContext). Regular HealthContributors that do not check against
a reactive API are executed on the elastic scheduler.
To provide custom health information from a reactive API, you can register Spring beans that
implement the ReactiveHealthIndicator interface. The following code shows a sample
ReactiveHealthIndicator implementation:
458
Java
import reactor.core.publisher.Mono;
import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.ReactiveHealthIndicator;
import org.springframework.stereotype.Component;
@Component
public class MyReactiveHealthIndicator implements ReactiveHealthIndicator {
@Override
public Mono<Health> health() {
return doHealthCheck().onErrorResume((exception) ->
Mono.just(new Health.Builder().down(exception).build()));
}
Kotlin
import org.springframework.boot.actuate.health.Health
import org.springframework.boot.actuate.health.ReactiveHealthIndicator
import org.springframework.stereotype.Component
import reactor.core.publisher.Mono
@Component
class MyReactiveHealthIndicator : ReactiveHealthIndicator {
459
Auto-configured ReactiveHealthIndicators
If necessary, reactive indicators replace the regular ones. Also, any HealthIndicator
TIP
that is not handled explicitly is wrapped automatically.
Health Groups
It is sometimes useful to organize health indicators into groups that you can use for different
purposes.
Properties
management.endpoint.health.group.custom.include=db
Yaml
management:
endpoint:
health:
group:
custom:
include: "db"
Similarly, to create a group that excludes the database indicators from the group and includes all
the other indicators, you can define the following:
460
Properties
management.endpoint.health.group.custom.exclude=db
Yaml
management:
endpoint:
health:
group:
custom:
exclude: "db"
By default, startup will fail if a health group includes or excludes a health indicator that does not
exist. To disable this behavior set management.endpoint.health.validate-group-membership to false.
By default, groups inherit the same StatusAggregator and HttpCodeStatusMapper settings as the
system health. However, you can also define these on a per-group basis. You can also override the
show-details and roles properties if required:
Properties
management.endpoint.health.group.custom.show-details=when-authorized
management.endpoint.health.group.custom.roles=admin
management.endpoint.health.group.custom.status.order=fatal,up
management.endpoint.health.group.custom.status.http-mapping.fatal=500
management.endpoint.health.group.custom.status.http-mapping.out-of-service=500
Yaml
management:
endpoint:
health:
group:
custom:
show-details: "when-authorized"
roles: "admin"
status:
order: "fatal,up"
http-mapping:
fatal: 500
out-of-service: 500
A health group can also include/exclude a CompositeHealthContributor. You can also include/exclude
only a certain component of a CompositeHealthContributor. This can be done using the fully
461
qualified name of the component as follows:
management.endpoint.health.group.custom.include="test/primary"
management.endpoint.health.group.custom.exclude="test/primary/b"
In the example above, the custom group will include the HealthContributor with the name primary
which is a component of the composite test. Here, primary itself is a composite and the
HealthContributor with the name b will be excluded from the custom group.
Health groups can be made available at an additional path on either the main or management port.
This is useful in cloud environments such as Kubernetes, where it is quite common to use a
separate management port for the actuator endpoints for security purposes. Having a separate port
could lead to unreliable health checks because the main application might not work properly even
if the health check is successful. The health group can be configured with an additional path as
follows:
management.endpoint.health.group.live.additional-path="server:/healthz"
This would make the live health group available on the main server port at /healthz. The prefix is
mandatory and must be either server: (represents the main server port) or management: (represents
the management port, if configured.) The path must be a single path segment.
DataSource Health
The DataSource health indicator shows the health of both standard data sources and routing data
source beans. The health of a routing data source includes the health of each of its target data
sources. In the health endpoint’s response, each of a routing data source’s targets is named by using
its routing key. If you prefer not to include routing data sources in the indicator’s output, set
management.health.db.ignore-routing-data-sources to true.
Applications deployed on Kubernetes can provide information about their internal state with
Container Probes. Depending on your Kubernetes configuration, the kubelet calls those probes and
reacts to the result.
By default, Spring Boot manages your Application Availability State. If deployed in a Kubernetes
environment, actuator gathers the “Liveness” and “Readiness” information from the
ApplicationAvailability interface and uses that information in dedicated health indicators:
LivenessStateHealthIndicator and ReadinessStateHealthIndicator. These indicators are shown on
the global health endpoint ("/actuator/health"). They are also exposed as separate HTTP Probes by
using health groups: "/actuator/health/liveness" and "/actuator/health/readiness".
You can then configure your Kubernetes infrastructure with the following endpoint information:
462
livenessProbe:
httpGet:
path: "/actuator/health/liveness"
port: <actuator-port>
failureThreshold: ...
periodSeconds: ...
readinessProbe:
httpGet:
path: "/actuator/health/readiness"
port: <actuator-port>
failureThreshold: ...
periodSeconds: ...
<actuator-port> should be set to the port that the actuator endpoints are available
NOTE on. It could be the main web server port or a separate management port if the
"management.server.port" property has been set.
These health groups are automatically enabled only if the application runs in a Kubernetes
environment. You can enable them in any environment by using the
management.endpoint.health.probes.enabled configuration property.
If your Actuator endpoints are deployed on a separate management context, the endpoints do not
use the same web infrastructure (port, connection pools, framework components) as the main
application. In this case, a probe check could be successful even if the main application does not
work properly (for example, it cannot accept new connections). For this reason, is it a good idea to
make the liveness and readiness health groups available on the main server port. This can be done
by setting the following property:
management.endpoint.health.probes.add-additional-paths=true
This would make liveness available at /livez and readiness at readyz on the main server port.
Actuator configures the “liveness” and “readiness” probes as Health Groups. This means that all the
health groups features are available for them. You can, for example, configure additional Health
463
Indicators:
Properties
management.endpoint.health.group.readiness.include=readinessState,customCheck
Yaml
management:
endpoint:
health:
group:
readiness:
include: "readinessState,customCheck"
By default, Spring Boot does not add other health indicators to these groups.
The “liveness” probe should not depend on health checks for external systems. If the liveness state
of an application is broken, Kubernetes tries to solve that problem by restarting the application
instance. This means that if an external system (such as a database, a Web API, or an external
cache) fails, Kubernetes might restart all application instances and create cascading failures.
As for the “readiness” probe, the choice of checking external systems must be made carefully by the
application developers. For this reason, Spring Boot does not include any additional health checks
in the readiness probe. If the readiness state of an application instance is unready, Kubernetes does
not route traffic to that instance. Some external systems might not be shared by application
instances, in which case they could be included in a readiness probe. Other external systems might
not be essential to the application (the application could have circuit breakers and fallbacks), in
which case they definitely should not be included. Unfortunately, an external system that is shared
by all application instances is common, and you have to make a judgement call: Include it in the
readiness probe and expect that the application is taken out of service when the external service is
down or leave it out and deal with failures higher up the stack, perhaps by using a circuit breaker
in the caller.
Also, if an application uses Kubernetes autoscaling, it may react differently to applications being
taken out of the load-balancer, depending on its autoscaler configuration.
464
Application Lifecycle and Probe States
An important aspect of the Kubernetes Probes support is its consistency with the application
lifecycle. There is a significant difference between the AvailabilityState (which is the in-memory,
internal state of the application) and the actual probe (which exposes that state). Depending on the
phase of application lifecycle, the probe might not be available.
Spring Boot publishes application events during startup and shutdown, and probes can listen to
such events and expose the AvailabilityState information.
The following tables show the AvailabilityState and the state of HTTP connectors at different
stages.
Ready CORRECT ACCEPTING_TR Accepts requests Startup tasks are finished. The
AFFIC application is receiving traffic.
Shutdown N/A N/A Server is shut down The application context is closed
complete and the application is shut down.
See Kubernetes container lifecycle section for more information about Kubernetes
TIP
deployment.
Application information exposes various information collected from all InfoContributor beans
defined in your ApplicationContext. Spring Boot includes a number of auto-configured
InfoContributor beans, and you can write your own.
465
Auto-configured InfoContributors
With no prerequisites to indicate that they should be enabled, the env, java, and os contributors are
disabled by default. Each can be enabled by setting its management.info.<id>.enabled property to
true.
The build and git info contributors are enabled by default. Each can be disabled by setting its
management.info.<id>.enabled property to false. Alternatively, to disable every contributor that is
usually enabled by default, set the management.info.defaults.enabled property to false.
When the env contributor is enabled, you can customize the data exposed by the info endpoint by
setting info.* Spring properties. All Environment properties under the info key are automatically
exposed. For example, you could add the following settings to your application.properties file:
Properties
info.app.encoding=UTF-8
info.app.java.source=17
info.app.java.target=17
Yaml
info:
app:
encoding: "UTF-8"
java:
source: "17"
target: "17"
466
Rather than hardcoding those values, you could also expand info properties at build
time.
Assuming you use Maven, you could rewrite the preceding example as follows:
Properties
info.app.encoding=@project.build.sourceEncoding@
info.app.java.source=@java.version@
info.app.java.target=@java.version@
TIP
Yaml
info:
app:
encoding: "@project.build.sourceEncoding@"
java:
source: "@java.version@"
target: "@java.version@"
Another useful feature of the info endpoint is its ability to publish information about the state of
your git source code repository when the project was built. If a GitProperties bean is available, you
can use the info endpoint to expose these properties.
Properties
management.info.git.mode=full
Yaml
management:
info:
git:
mode: "full"
To disable the git commit information from the info endpoint completely, set the
management.info.git.enabled property to false, as follows:
467
Properties
management.info.git.enabled=false
Yaml
management:
info:
git:
enabled: false
Build Information
If a BuildProperties bean is available, the info endpoint can also publish information about your
build. This happens if a META-INF/build-info.properties file is available in the classpath.
The Maven and Gradle plugins can both generate that file. See "how to generate build
TIP
information" for more details.
Java Information
The info endpoint publishes information about your Java runtime environment, see JavaInfo for
more details.
OS Information
The info endpoint publishes information about your Operating System, see OsInfo for more details.
To provide custom application information, you can register Spring beans that implement the
InfoContributor interface.
468
Java
import java.util.Collections;
import org.springframework.boot.actuate.info.Info;
import org.springframework.boot.actuate.info.InfoContributor;
import org.springframework.stereotype.Component;
@Component
public class MyInfoContributor implements InfoContributor {
@Override
public void contribute(Info.Builder builder) {
builder.withDetail("example", Collections.singletonMap("key", "value"));
}
Kotlin
import org.springframework.boot.actuate.info.Info
import org.springframework.boot.actuate.info.InfoContributor
import org.springframework.stereotype.Component
import java.util.Collections
@Component
class MyInfoContributor : InfoContributor {
If you reach the info endpoint, you should see a response that contains the following additional
entry:
{
"example": {
"key" : "value"
}
}
469
Actuator is supported natively with Spring MVC, Spring WebFlux, and Jersey. If both
TIP
Jersey and Spring MVC are available, Spring MVC is used.
Sometimes, it is useful to customize the prefix for the management endpoints. For example, your
application might already use /actuator for another purpose. You can use the
management.endpoints.web.base-path property to change the prefix for your management endpoint,
as the following example shows:
Properties
management.endpoints.web.base-path=/manage
Yaml
management:
endpoints:
web:
base-path: "/manage"
Unless the management port has been configured to expose endpoints by using a
different HTTP port, management.endpoints.web.base-path is relative to
NOTE server.servlet.context-path (for servlet web applications) or spring.webflux.base-
path (for reactive web applications). If management.server.port is configured,
management.endpoints.web.base-path is relative to management.server.base-path.
If you want to map endpoints to a different path, you can use the management.endpoints.web.path-
mapping property.
Properties
management.endpoints.web.base-path=/
management.endpoints.web.path-mapping.health=healthcheck
470
Yaml
management:
endpoints:
web:
base-path: "/"
path-mapping:
health: "healthcheck"
Exposing management endpoints by using the default HTTP port is a sensible choice for cloud-
based deployments. If, however, your application runs inside your own data center, you may prefer
to expose endpoints by using a different HTTP port.
You can set the management.server.port property to change the HTTP port, as the following example
shows:
Properties
management.server.port=8081
Yaml
management:
server:
port: 8081
On Cloud Foundry, by default, applications receive requests only on port 8080 for
both HTTP and TCP routing. If you want to use a custom management port on Cloud
NOTE
Foundry, you need to explicitly set up the application’s routes to forward traffic to
the custom port.
When configured to use a custom port, you can also configure the management server with its own
SSL by using the various management.server.ssl.* properties. For example, doing so lets a
management server be available over HTTP while the main application uses HTTPS, as the
following property settings show:
471
Properties
server.port=8443
server.ssl.enabled=true
server.ssl.key-store=classpath:store.jks
server.ssl.key-password=secret
management.server.port=8080
management.server.ssl.enabled=false
Yaml
server:
port: 8443
ssl:
enabled: true
key-store: "classpath:store.jks"
key-password: "secret"
management:
server:
port: 8080
ssl:
enabled: false
Alternatively, both the main server and the management server can use SSL but with different key
stores, as follows:
Properties
server.port=8443
server.ssl.enabled=true
server.ssl.key-store=classpath:main.jks
server.ssl.key-password=secret
management.server.port=8080
management.server.ssl.enabled=true
management.server.ssl.key-store=classpath:management.jks
management.server.ssl.key-password=secret
472
Yaml
server:
port: 8443
ssl:
enabled: true
key-store: "classpath:main.jks"
key-password: "secret"
management:
server:
port: 8080
ssl:
enabled: true
key-store: "classpath:management.jks"
key-password: "secret"
You can customize the address on which the management endpoints are available by setting the
management.server.address property. Doing so can be useful if you want to listen only on an internal
or ops-facing network or to listen only for connections from localhost.
You can listen on a different address only when the port differs from the main
NOTE
server port.
The following example application.properties does not allow remote management connections:
Properties
management.server.port=8081
management.server.address=127.0.0.1
Yaml
management:
server:
port: 8081
address: "127.0.0.1"
If you do not want to expose endpoints over HTTP, you can set the management port to -1, as the
following example shows:
Properties
management.server.port=-1
473
Yaml
management:
server:
port: -1
You can also achieve this by using the management.endpoints.web.exposure.exclude property, as the
following example shows:
Properties
management.endpoints.web.exposure.exclude=*
Yaml
management:
endpoints:
web:
exposure:
exclude: "*"
If your platform provides a standard MBeanServer, Spring Boot uses that and defaults to the VM
MBeanServer, if necessary. If all that fails, a new MBeanServer is created.
By default, Spring Boot also exposes management endpoints as JMX MBeans under the
org.springframework.boot domain. To take full control over endpoint registration in the JMX
domain, consider registering your own EndpointObjectNameFactory implementation.
The name of the MBean is usually generated from the id of the endpoint. For example, the health
endpoint is exposed as org.springframework.boot:type=Endpoint,name=Health.
If your application contains more than one Spring ApplicationContext, you may find that names
clash. To solve this problem, you can set the spring.jmx.unique-names property to true so that MBean
names are always unique.
You can also customize the JMX domain under which endpoints are exposed. The following settings
474
show an example of doing so in application.properties:
Properties
spring.jmx.unique-names=true
management.endpoints.jmx.domain=com.example.myapp
Yaml
spring:
jmx:
unique-names: true
management:
endpoints:
jmx:
domain: "com.example.myapp"
If you do not want to expose endpoints over JMX, you can set the
management.endpoints.jmx.exposure.exclude property to *, as the following example shows:
Properties
management.endpoints.jmx.exposure.exclude=*
Yaml
management:
endpoints:
jmx:
exposure:
exclude: "*"
13.5. Observability
Observability is the ability to observe the internal state of a running system from the outside. It
consists of the three pillars logging, metrics and traces.
For metrics and traces, Spring Boot uses Micrometer Observation. To create your own observations
(which will lead to metrics and traces), you can inject an ObservationRegistry.
475
import io.micrometer.observation.Observation;
import io.micrometer.observation.ObservationRegistry;
import org.springframework.stereotype.Component;
@Component
public class MyCustomObservation {
Low cardinality tags will be added to metrics and traces, while high cardinality tags
NOTE
will only be added to traces.
Observability for JDBC and R2DBC can be configured using separate projects. For
JDBC, the Datasource Micrometer project provides a Spring Boot starter which
automatically creates observations when JDBC operations are invoked. Read more
TIP
about it in the reference documentation. For R2DBC, the Spring Boot Auto
Configuration for R2DBC Observation creates observations for R2DBC query
invocations.
The next sections will provide more details about logging, metrics and traces.
13.6. Loggers
Spring Boot Actuator includes the ability to view and configure the log levels of your application at
runtime. You can view either the entire list or an individual logger’s configuration, which is made
up of both the explicitly configured logging level as well as the effective logging level given to it by
476
the logging framework. These levels can be one of:
• TRACE
• DEBUG
• INFO
• WARN
• ERROR
• FATAL
• OFF
• null
To configure a given logger, POST a partial entity to the resource’s URI, as the following example
shows:
{
"configuredLevel": "DEBUG"
}
To “reset” the specific level of the logger (and use the default configuration instead),
TIP
you can pass a value of null as the configuredLevel.
13.7. Metrics
Spring Boot Actuator provides dependency management and auto-configuration for Micrometer, an
application metrics facade that supports numerous monitoring systems, including:
• AppOptics
• Atlas
• Datadog
• Dynatrace
• Elastic
• Ganglia
• Graphite
• Humio
• Influx
• JMX
477
• KairosDB
• New Relic
• OpenTelemetry
• Prometheus
• SignalFx
• Simple (in-memory)
• Stackdriver
• StatsD
• Wavefront
Spring Boot auto-configures a composite MeterRegistry and adds a registry to the composite for
each of the supported implementations that it finds on the classpath. Having a dependency on
micrometer-registry-{system} in your runtime classpath is enough for Spring Boot to configure the
registry.
Most registries share common features. For instance, you can disable a particular registry even if
the Micrometer registry implementation is on the classpath. The following example disables
Datadog:
Properties
management.datadog.metrics.export.enabled=false
Yaml
management:
datadog:
metrics:
export:
enabled: false
You can also disable all registries unless stated otherwise by the registry-specific property, as the
following example shows:
Properties
management.defaults.metrics.export.enabled=false
478
Yaml
management:
defaults:
metrics:
export:
enabled: false
Spring Boot also adds any auto-configured registries to the global static composite registry on the
Metrics class, unless you explicitly tell it not to:
Properties
management.metrics.use-global-registry=false
Yaml
management:
metrics:
use-global-registry: false
You can register any number of MeterRegistryCustomizer beans to further configure the registry,
such as applying common tags, before any meters are registered with the registry:
Java
import io.micrometer.core.instrument.MeterRegistry;
import org.springframework.boot.actuate.autoconfigure.metrics.MeterRegistryCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyMeterRegistryConfiguration {
@Bean
public MeterRegistryCustomizer<MeterRegistry> metricsCommonTags() {
return (registry) -> registry.config().commonTags("region", "us-east-1");
}
479
Kotlin
import io.micrometer.core.instrument.MeterRegistry
import org.springframework.boot.actuate.autoconfigure.metrics.MeterRegistryCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyMeterRegistryConfiguration {
@Bean
fun metricsCommonTags(): MeterRegistryCustomizer<MeterRegistry> {
return MeterRegistryCustomizer { registry ->
registry.config().commonTags("region", "us-east-1")
}
}
You can apply customizations to particular registry implementations by being more specific about
the generic type:
Java
import io.micrometer.core.instrument.Meter;
import io.micrometer.core.instrument.config.NamingConvention;
import io.micrometer.graphite.GraphiteMeterRegistry;
import org.springframework.boot.actuate.autoconfigure.metrics.MeterRegistryCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyMeterRegistryConfiguration {
@Bean
public MeterRegistryCustomizer<GraphiteMeterRegistry>
graphiteMetricsNamingConvention() {
return (registry) -> registry.config().namingConvention(this::name);
}
480
Kotlin
import io.micrometer.core.instrument.Meter
import io.micrometer.core.instrument.config.NamingConvention
import io.micrometer.graphite.GraphiteMeterRegistry
import org.springframework.boot.actuate.autoconfigure.metrics.MeterRegistryCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyMeterRegistryConfiguration {
@Bean
fun graphiteMetricsNamingConvention():
MeterRegistryCustomizer<GraphiteMeterRegistry> {
return MeterRegistryCustomizer { registry: GraphiteMeterRegistry ->
registry.config().namingConvention(this::name)
}
}
Spring Boot also configures built-in instrumentation that you can control through configuration or
dedicated annotation markers.
AppOptics
Properties
management.appoptics.metrics.export.api-token=YOUR_TOKEN
Yaml
management:
appoptics:
metrics:
export:
api-token: "YOUR_TOKEN"
481
Atlas
By default, metrics are exported to Atlas running on your local machine. You can provide the
location of the Atlas server:
Properties
management.atlas.metrics.export.uri=https://atlas.example.com:7101/api/v1/publish
Yaml
management:
atlas:
metrics:
export:
uri: "https://atlas.example.com:7101/api/v1/publish"
Datadog
A Datadog registry periodically pushes metrics to datadoghq. To export metrics to Datadog, you
must provide your API key:
Properties
management.datadog.metrics.export.api-key=YOUR_KEY
Yaml
management:
datadog:
metrics:
export:
api-key: "YOUR_KEY"
If you additionally provide an application key (optional), then metadata such as meter descriptions,
types, and base units will also be exported:
Properties
management.datadog.metrics.export.api-key=YOUR_API_KEY
management.datadog.metrics.export.application-key=YOUR_APPLICATION_KEY
482
Yaml
management:
datadog:
metrics:
export:
api-key: "YOUR_API_KEY"
application-key: "YOUR_APPLICATION_KEY"
By default, metrics are sent to the Datadog US site (api.datadoghq.com). If your Datadog project is
hosted on one of the other sites, or you need to send metrics through a proxy, configure the URI
accordingly:
Properties
management.datadog.metrics.export.uri=https://api.datadoghq.eu
Yaml
management:
datadog:
metrics:
export:
uri: "https://api.datadoghq.eu"
You can also change the interval at which metrics are sent to Datadog:
Properties
management.datadog.metrics.export.step=30s
Yaml
management:
datadog:
metrics:
export:
step: "30s"
Dynatrace
Dynatrace offers two metrics ingest APIs, both of which are implemented for Micrometer. You can
find the Dynatrace documentation on Micrometer metrics ingest here. Configuration properties in
the v1 namespace apply only when exporting to the Timeseries v1 API. Configuration properties in
the v2 namespace apply only when exporting to the Metrics v2 API. Note that this integration can
export only to either the v1 or v2 version of the API at a time, with v2 being preferred. If the device-
id (required for v1 but not used in v2) is set in the v1 namespace, metrics are exported to the v1
endpoint. Otherwise, v2 is assumed.
483
v2 API
Auto-configuration
Dynatrace auto-configuration is available for hosts that are monitored by the OneAgent or by the
Dynatrace Operator for Kubernetes.
Local OneAgent: If a OneAgent is running on the host, metrics are automatically exported to the
local OneAgent ingest endpoint. The ingest endpoint forwards the metrics to the Dynatrace
backend.
Dynatrace Kubernetes Operator: When running in Kubernetes with the Dynatrace Operator
installed, the registry will automatically pick up your endpoint URI and API token from the
operator instead.
This is the default behavior and requires no special setup beyond a dependency on
io.micrometer:micrometer-registry-dynatrace.
Manual configuration
If no auto-configuration is available, the endpoint of the Metrics v2 API and an API token are
required. The API token must have the “Ingest metrics” (metrics.ingest) permission set. We
recommend limiting the scope of the token to this one permission. You must ensure that the
endpoint URI contains the path (for example, /api/v2/metrics/ingest):
The URL of the Metrics API v2 ingest endpoint is different according to your deployment option:
• SaaS: https://{your-environment-id}.live.dynatrace.com/api/v2/metrics/ingest
The example below configures metrics export using the example environment id:
Properties
management.dynatrace.metrics.export.uri=https://example.live.dynatrace.com/api/v2/metr
ics/ingest
management.dynatrace.metrics.export.api-token=YOUR_TOKEN
Yaml
management:
dynatrace:
metrics:
export:
uri: "https://example.live.dynatrace.com/api/v2/metrics/ingest"
api-token: "YOUR_TOKEN"
When using the Dynatrace v2 API, the following optional features are available (more details can be
484
found in the Dynatrace documentation):
• Metric key prefix: Sets a prefix that is prepended to all exported metric keys.
• Default dimensions: Specify key-value pairs that are added to all exported metrics. If tags with
the same key are specified with Micrometer, they overwrite the default dimensions.
• Use Dynatrace Summary instruments: In some cases the Micrometer Dynatrace registry created
metrics that were rejected. In Micrometer 1.9.x, this was fixed by introducing Dynatrace-specific
summary instruments. Setting this toggle to false forces Micrometer to fall back to the behavior
that was the default before 1.9.x. It should only be used when encountering problems while
migrating from Micrometer 1.8.x to 1.9.x.
It is possible to not specify a URI and API token, as shown in the following example. In this scenario,
the automatically configured endpoint is used:
Properties
management.dynatrace.metrics.export.v2.metric-key-prefix=your.key.prefix
management.dynatrace.metrics.export.v2.enrich-with-dynatrace-metadata=true
management.dynatrace.metrics.export.v2.default-dimensions.key1=value1
management.dynatrace.metrics.export.v2.default-dimensions.key2=value2
management.dynatrace.metrics.export.v2.use-dynatrace-summary-instruments=true
Yaml
management:
dynatrace:
metrics:
export:
# Specify uri and api-token here if not using the local OneAgent endpoint.
v2:
metric-key-prefix: "your.key.prefix"
enrich-with-dynatrace-metadata: true
default-dimensions:
key1: "value1"
key2: "value2"
use-dynatrace-summary-instruments: true # (default: true)
v1 API (Legacy)
The Dynatrace v1 API metrics registry pushes metrics to the configured URI periodically by using
the Timeseries v1 API. For backwards-compatibility with existing setups, when device-id is set
(required for v1, but not used in v2), metrics are exported to the Timeseries v1 endpoint. To export
metrics to Dynatrace, your API token, device ID, and URI must be provided:
485
Properties
management.dynatrace.metrics.export.uri=https://{your-environment-
id}.live.dynatrace.com
management.dynatrace.metrics.export.api-token=YOUR_TOKEN
management.dynatrace.metrics.export.v1.device-id=YOUR_DEVICE_ID
Yaml
management:
dynatrace:
metrics:
export:
uri: "https://{your-environment-id}.live.dynatrace.com"
api-token: "YOUR_TOKEN"
v1:
device-id: "YOUR_DEVICE_ID"
For the v1 API, you must specify the base environment URI without a path, as the v1 endpoint path
is added automatically.
Version-independent Settings
In addition to the API endpoint and token, you can also change the interval at which metrics are
sent to Dynatrace. The default export interval is 60s. The following example sets the export interval
to 30 seconds:
Properties
management.dynatrace.metrics.export.step=30s
Yaml
management:
dynatrace:
metrics:
export:
step: "30s"
You can find more information on how to set up the Dynatrace exporter for Micrometer in the
Micrometer documentation and the Dynatrace documentation.
Elastic
By default, metrics are exported to Elastic running on your local machine. You can provide the
location of the Elastic server to use by using the following property:
486
Properties
management.elastic.metrics.export.host=https://elastic.example.com:8086
Yaml
management:
elastic:
metrics:
export:
host: "https://elastic.example.com:8086"
Ganglia
By default, metrics are exported to Ganglia running on your local machine. You can provide the
Ganglia server host and port, as the following example shows:
Properties
management.ganglia.metrics.export.host=ganglia.example.com
management.ganglia.metrics.export.port=9649
Yaml
management:
ganglia:
metrics:
export:
host: "ganglia.example.com"
port: 9649
Graphite
By default, metrics are exported to Graphite running on your local machine. You can provide the
Graphite server host and port, as the following example shows:
Properties
management.graphite.metrics.export.host=graphite.example.com
management.graphite.metrics.export.port=9004
487
Yaml
management:
graphite:
metrics:
export:
host: "graphite.example.com"
port: 9004
To take control over this behavior, define your GraphiteMeterRegistry and supply your
own HierarchicalNameMapper. An auto-configured GraphiteConfig and Clock beans are
provided unless you define your own:
Java
import io.micrometer.core.instrument.Clock;
import io.micrometer.core.instrument.Meter;
import io.micrometer.core.instrument.config.NamingConvention;
import io.micrometer.core.instrument.util.HierarchicalNameMapper;
import io.micrometer.graphite.GraphiteConfig;
import io.micrometer.graphite.GraphiteMeterRegistry;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyGraphiteConfiguration {
TIP
@Bean
public GraphiteMeterRegistry graphiteMeterRegistry(GraphiteConfig
config, Clock clock) {
return new GraphiteMeterRegistry(config, clock,
this::toHierarchicalName);
}
488
Kotlin
import io.micrometer.core.instrument.Clock
import io.micrometer.core.instrument.Meter
import io.micrometer.core.instrument.config.NamingConvention
import io.micrometer.core.instrument.util.HierarchicalNameMapper
import io.micrometer.graphite.GraphiteConfig
import io.micrometer.graphite.GraphiteMeterRegistry
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyGraphiteConfiguration {
@Bean
fun graphiteMeterRegistry(config: GraphiteConfig, clock: Clock):
GraphiteMeterRegistry {
return GraphiteMeterRegistry(config, clock,
this::toHierarchicalName)
}
private fun toHierarchicalName(id: Meter.Id, convention:
NamingConvention): String {
return ...
}
Humio
By default, the Humio registry periodically pushes metrics to cloud.humio.com. To export metrics to
SaaS Humio, you must provide your API token:
Properties
management.humio.metrics.export.api-token=YOUR_TOKEN
Yaml
management:
humio:
metrics:
export:
api-token: "YOUR_TOKEN"
You should also configure one or more tags to identify the data source to which metrics are pushed:
489
Properties
management.humio.metrics.export.tags.alpha=a
management.humio.metrics.export.tags.bravo=b
Yaml
management:
humio:
metrics:
export:
tags:
alpha: "a"
bravo: "b"
Influx
By default, metrics are exported to an Influx v1 instance running on your local machine with the
default configuration. To export metrics to InfluxDB v2, configure the org, bucket, and
authentication token for writing metrics. You can provide the location of the Influx server to use by
using:
Properties
management.influx.metrics.export.uri=https://influx.example.com:8086
Yaml
management:
influx:
metrics:
export:
uri: "https://influx.example.com:8086"
JMX
Micrometer provides a hierarchical mapping to JMX, primarily as a cheap and portable way to view
metrics locally. By default, metrics are exported to the metrics JMX domain. You can provide the
domain to use by using:
Properties
management.jmx.metrics.export.domain=com.example.app.metrics
490
Yaml
management:
jmx:
metrics:
export:
domain: "com.example.app.metrics"
To take control over this behavior, define your JmxMeterRegistry and supply your own
HierarchicalNameMapper. An auto-configured JmxConfig and Clock beans are provided
unless you define your own:
Java
import io.micrometer.core.instrument.Clock;
import io.micrometer.core.instrument.Meter;
import io.micrometer.core.instrument.config.NamingConvention;
import io.micrometer.core.instrument.util.HierarchicalNameMapper;
import io.micrometer.jmx.JmxConfig;
import io.micrometer.jmx.JmxMeterRegistry;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyJmxConfiguration {
TIP @Bean
public JmxMeterRegistry jmxMeterRegistry(JmxConfig config, Clock
clock) {
return new JmxMeterRegistry(config, clock,
this::toHierarchicalName);
}
491
Kotlin
import io.micrometer.core.instrument.Clock
import io.micrometer.core.instrument.Meter
import io.micrometer.core.instrument.config.NamingConvention
import io.micrometer.core.instrument.util.HierarchicalNameMapper
import io.micrometer.jmx.JmxConfig
import io.micrometer.jmx.JmxMeterRegistry
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyJmxConfiguration {
@Bean
fun jmxMeterRegistry(config: JmxConfig, clock: Clock):
JmxMeterRegistry {
return JmxMeterRegistry(config, clock, this::toHierarchicalName)
}
KairosDB
By default, metrics are exported to KairosDB running on your local machine. You can provide the
location of the KairosDB server to use by using:
Properties
management.kairos.metrics.export.uri=https://kairosdb.example.com:8080/api/v1/datapoin
ts
Yaml
management:
kairos:
metrics:
export:
uri: "https://kairosdb.example.com:8080/api/v1/datapoints"
New Relic
A New Relic registry periodically pushes metrics to New Relic. To export metrics to New Relic, you
must provide your API key and account ID:
492
Properties
management.newrelic.metrics.export.api-key=YOUR_KEY
management.newrelic.metrics.export.account-id=YOUR_ACCOUNT_ID
Yaml
management:
newrelic:
metrics:
export:
api-key: "YOUR_KEY"
account-id: "YOUR_ACCOUNT_ID"
You can also change the interval at which metrics are sent to New Relic:
Properties
management.newrelic.metrics.export.step=30s
Yaml
management:
newrelic:
metrics:
export:
step: "30s"
By default, metrics are published through REST calls, but you can also use the Java Agent API if you
have it on the classpath:
Properties
management.newrelic.metrics.export.client-provider-type=insights-agent
Yaml
management:
newrelic:
metrics:
export:
client-provider-type: "insights-agent"
Finally, you can take full control by defining your own NewRelicClientProvider bean.
493
OpenTelemetry
By default, metrics are exported to OpenTelemetry running on your local machine. You can provide
the location of the OpenTelemetry metric endpoint to use by using:
Properties
management.otlp.metrics.export.url=https://otlp.example.com:4318/v1/metrics
Yaml
management:
otlp:
metrics:
export:
url: "https://otlp.example.com:4318/v1/metrics"
Prometheus
Prometheus expects to scrape or poll individual application instances for metrics. Spring Boot
provides an actuator endpoint at /actuator/prometheus to present a Prometheus scrape with the
appropriate format.
By default, the endpoint is not available and must be exposed. See exposing endpoints
TIP
for more details.
scrape_configs:
- job_name: "spring"
metrics_path: "/actuator/prometheus"
static_configs:
- targets: ["HOST:PORT"]
Prometheus Exemplars are also supported. To enable this feature, a SpanContextSupplier bean
should be present. If you use Micrometer Tracing, this will be auto-configured for you, but you can
always create your own if you want. Please check the Prometheus Docs, since this feature needs to
be explicitly enabled on Prometheus' side, and it is only supported using the OpenMetrics format.
For ephemeral or batch jobs that may not exist long enough to be scraped, you can use Prometheus
Pushgateway support to expose the metrics to Prometheus. To enable Prometheus Pushgateway
support, add the following dependency to your project:
<dependency>
<groupId>io.prometheus</groupId>
<artifactId>simpleclient_pushgateway</artifactId>
</dependency>
494
When the Prometheus Pushgateway dependency is present on the classpath and the
management.prometheus.metrics.export.pushgateway.enabled property is set to true, a
PrometheusPushGatewayManager bean is auto-configured. This manages the pushing of metrics to a
Prometheus Pushgateway.
SignalFx
SignalFx registry periodically pushes metrics to SignalFx. To export metrics to SignalFx, you must
provide your access token:
Properties
management.signalfx.metrics.export.access-token=YOUR_ACCESS_TOKEN
Yaml
management:
signalfx:
metrics:
export:
access-token: "YOUR_ACCESS_TOKEN"
You can also change the interval at which metrics are sent to SignalFx:
Properties
management.signalfx.metrics.export.step=30s
Yaml
management:
signalfx:
metrics:
export:
step: "30s"
Simple
Micrometer ships with a simple, in-memory backend that is automatically used as a fallback if no
other registry is configured. This lets you see what metrics are collected in the metrics endpoint.
The in-memory backend disables itself as soon as you use any other available backend. You can also
disable it explicitly:
495
Properties
management.simple.metrics.export.enabled=false
Yaml
management:
simple:
metrics:
export:
enabled: false
Stackdriver
The Stackdriver registry periodically pushes metrics to Stackdriver. To export metrics to SaaS
Stackdriver, you must provide your Google Cloud project ID:
Properties
management.stackdriver.metrics.export.project-id=my-project
Yaml
management:
stackdriver:
metrics:
export:
project-id: "my-project"
You can also change the interval at which metrics are sent to Stackdriver:
Properties
management.stackdriver.metrics.export.step=30s
Yaml
management:
stackdriver:
metrics:
export:
step: "30s"
StatsD
The StatsD registry eagerly pushes metrics over UDP to a StatsD agent. By default, metrics are
exported to a StatsD agent running on your local machine. You can provide the StatsD agent host,
496
port, and protocol to use by using:
Properties
management.statsd.metrics.export.host=statsd.example.com
management.statsd.metrics.export.port=9125
management.statsd.metrics.export.protocol=udp
Yaml
management:
statsd:
metrics:
export:
host: "statsd.example.com"
port: 9125
protocol: "udp"
You can also change the StatsD line protocol to use (it defaults to Datadog):
Properties
management.statsd.metrics.export.flavor=etsy
Yaml
management:
statsd:
metrics:
export:
flavor: "etsy"
Wavefront
The Wavefront registry periodically pushes metrics to Wavefront. If you are exporting metrics to
Wavefront directly, you must provide your API token:
Properties
management.wavefront.api-token=YOUR_API_TOKEN
Yaml
management:
wavefront:
api-token: "YOUR_API_TOKEN"
Alternatively, you can use a Wavefront sidecar or an internal proxy in your environment to
497
forward metrics data to the Wavefront API host:
Properties
management.wavefront.uri=proxy://localhost:2878
Yaml
management:
wavefront:
uri: "proxy://localhost:2878"
You can also change the interval at which metrics are sent to Wavefront:
Properties
management.wavefront.metrics.export.step=30s
Yaml
management:
wavefront:
metrics:
export:
step: "30s"
Spring Boot provides automatic meter registration for a wide variety of technologies. In most
situations, the defaults provide sensible metrics that can be published to any of the supported
monitoring systems.
JVM Metrics
Auto-configuration enables JVM Metrics by using core Micrometer classes. JVM metrics are
published under the jvm. meter name.
• Thread utilization
498
• JVM version information
System Metrics
Auto-configuration enables system metrics by using core Micrometer classes. System metrics are
published under the system., process., and disk. meter names.
• CPU metrics
• Uptime metrics (both the amount of time the application has been running and a fixed gauge of
the absolute start time)
Metrics are tagged by the fully qualified name of the application class.
Logger Metrics
Auto-configuration enables the event metrics for both Logback and Log4J2. The details are
published under the log4j2.events. or logback.events. meter names.
Auto-configuration enables the instrumentation of all requests handled by Spring MVC controllers
and functional handlers. By default, metrics are generated with the name, http.server.requests.
You can customize the name by setting the management.observations.http.server.requests.name
property.
See the Spring Framework reference documentation for more information on produced
observations.
499
In some cases, exceptions handled in web controllers are not recorded as request
TIP metrics tags. Applications can opt in and record exceptions by setting handled
exceptions as request attributes.
By default, all requests are handled. To customize the filter, provide a @Bean that implements
FilterRegistrationBean<WebMvcMetricsFilter>.
See the Spring Framework reference documentation for more information on produced
observations.
In some cases, exceptions handled in controllers and handler functions are not
TIP recorded as request metrics tags. Applications can opt in and record exceptions by
setting handled exceptions as request attributes.
Auto-configuration enables the instrumentation of all requests handled by the Jersey JAX-RS
implementation. By default, metrics are generated with the name, http.server.requests. You can
customize the name by setting the management.observations.http.server.requests.name property.
By default, Jersey server metrics are tagged with the following information:
Tag Description
exception The simple class name of any exception that was
thrown while handling the request.
method The request’s method (for example, GET or POST)
outcome The request’s outcome, based on the status code
of the response. 1xx is INFORMATIONAL, 2xx is
SUCCESS, 3xx is REDIRECTION, 4xx is CLIENT_ERROR,
and 5xx is SERVER_ERROR
status The response’s HTTP status code (for example,
200 or 500)
uri The request’s URI template prior to variable
substitution, if possible (for example,
/api/person/{id})
500
To customize the tags, provide a @Bean that implements JerseyTagsProvider.
Spring Boot Actuator manages the instrumentation of both RestTemplate and WebClient. For that,
you have to inject the auto-configured builder and use it to create instances:
You can also manually apply the customizers responsible for this instrumentation, namely
ObservationRestTemplateCustomizer and ObservationWebClientCustomizer.
By default, metrics are generated with the name, http.client.requests. You can customize the name
by setting the management.observations.http.client.requests.name property.
See the Spring Framework reference documentation for more information on produced
observations.
To customize the tags when using RestTemplate, provide a @Bean that implements
ClientRequestObservationConvention from the org.springframework.http.client.observation
package. To customize the tags when using WebClient, provide a @Bean that implements
ClientRequestObservationConvention from the org.springframework.web.reactive.function.client
package.
Tomcat Metrics
Cache Metrics
Auto-configuration enables the instrumentation of all available Cache instances on startup, with
metrics prefixed with cache. Cache instrumentation is standardized for a basic set of metrics.
Additional, cache-specific metrics are also available.
• Cache2k
• Caffeine
• Hazelcast
• Redis
Metrics are tagged by the name of the cache and by the name of the CacheManager, which is derived
from the bean name.
501
Only caches that are configured on startup are bound to the registry. For caches not
defined in the cache’s configuration, such as caches created on the fly or
NOTE
programmatically after the startup phase, an explicit registration is required. A
CacheMetricsRegistrar bean is made available to make that process easier.
DataSource Metrics
Auto-configuration enables the instrumentation of all available DataSource objects with metrics
prefixed with jdbc.connections. Data source instrumentation results in gauges that represent the
currently active, idle, maximum allowed, and minimum allowed connections in the pool.
Metrics are also tagged by the name of the DataSource computed based on the bean name.
By default, Spring Boot provides metadata for all supported data sources. You can add
TIP additional DataSourcePoolMetadataProvider beans if your favorite data source is not
supported. See DataSourcePoolMetadataProvidersConfiguration for examples.
Also, Hikari-specific metrics are exposed with a hikaricp prefix. Each metric is tagged by the name
of the pool (you can control it with spring.datasource.name).
Hibernate Metrics
Metrics are also tagged by the name of the EntityManagerFactory, which is derived from the bean
name.
To enable statistics, the standard JPA property hibernate.generate_statistics must be set to true.
You can enable that on the auto-configured EntityManagerFactory:
Properties
spring.jpa.properties[hibernate.generate_statistics]=true
502
Yaml
spring:
jpa:
properties:
"[hibernate.generate_statistics]": true
Auto-configuration enables the instrumentation of all Spring Data Repository method invocations.
By default, metrics are generated with the name, spring.data.repository.invocations. You can
customize the name by setting the management.metrics.data.repository.metric-name property.
A @Timed annotation with longTask = true enables a long task timer for the method.
NOTE Long task timers require a separate metric name and can be stacked with a short
task timer.
By default, repository invocation related metrics are tagged with the following information:
Tag Description
repository The simple class name of the source Repository.
method The name of the Repository method that was
invoked.
state The result state (SUCCESS, ERROR, CANCELED, or
RUNNING).
exception The simple class name of any exception that was
thrown from the invocation.
RabbitMQ Metrics
Auto-configuration enables the instrumentation of all available RabbitMQ connection factories with
a metric named rabbitmq.
503
Kafka Metrics
MongoDB Metrics
A timer metric named mongodb.driver.commands is created for each command issued to the
underlying MongoDB driver. Each metric is tagged with the following information by default:
Tag Description
command The name of the command issued.
cluster.id The identifier of the cluster to which the
command was sent.
server.address The address of the server to which the command
was sent.
status The outcome of the command (SUCCESS or
FAILED).
To replace the default metric tags, define a MongoCommandTagsProvider bean, as the following
example shows:
Java
import io.micrometer.core.instrument.binder.mongodb.MongoCommandTagsProvider;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyCommandTagsProviderConfiguration {
@Bean
public MongoCommandTagsProvider customCommandTagsProvider() {
return new CustomCommandTagsProvider();
}
504
Kotlin
import io.micrometer.core.instrument.binder.mongodb.MongoCommandTagsProvider
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyCommandTagsProviderConfiguration {
@Bean
fun customCommandTagsProvider(): MongoCommandTagsProvider? {
return CustomCommandTagsProvider()
}
Properties
management.metrics.mongo.command.enabled=false
Yaml
management:
metrics:
mongo:
command:
enabled: false
The following gauge metrics are created for the connection pool:
• mongodb.driver.pool.size reports the current size of the connection pool, including idle and and
in-use members.
• mongodb.driver.pool.waitqueuesize reports the current size of the wait queue for a connection
from the pool.
Tag Description
cluster.id The identifier of the cluster to which the
connection pool corresponds.
505
Tag Description
server.address The address of the server to which the
connection pool corresponds.
Java
import io.micrometer.core.instrument.binder.mongodb.MongoConnectionPoolTagsProvider;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyConnectionPoolTagsProviderConfiguration {
@Bean
public MongoConnectionPoolTagsProvider customConnectionPoolTagsProvider() {
return new CustomConnectionPoolTagsProvider();
}
Kotlin
import io.micrometer.core.instrument.binder.mongodb.MongoConnectionPoolTagsProvider
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyConnectionPoolTagsProviderConfiguration {
@Bean
fun customConnectionPoolTagsProvider(): MongoConnectionPoolTagsProvider {
return CustomConnectionPoolTagsProvider()
}
To disable the auto-configured connection pool metrics, set the following property:
Properties
management.metrics.mongo.connectionpool.enabled=false
506
Yaml
management:
metrics:
mongo:
connectionpool:
enabled: false
Jetty Metrics
To use @Timed where it is not directly supported by Spring Boot, refer to the Micrometer
documentation.
Redis Metrics
Java
import io.micrometer.core.instrument.MeterRegistry;
import io.micrometer.core.instrument.Tags;
import org.springframework.stereotype.Component;
@Component
public class MyBean {
507
Kotlin
import io.micrometer.core.instrument.MeterRegistry
import io.micrometer.core.instrument.Tags
import org.springframework.stereotype.Component
@Component
class MyBean(registry: MeterRegistry) {
init {
dictionary = Dictionary.load()
registry.gauge("dictionary.size", Tags.empty(), dictionary.words.size)
}
If your metrics depend on other beans, we recommend that you use a MeterBinder to register them:
Java
import io.micrometer.core.instrument.Gauge;
import io.micrometer.core.instrument.binder.MeterBinder;
import org.springframework.context.annotation.Bean;
@Bean
public MeterBinder queueSize(Queue queue) {
return (registry) -> Gauge.builder("queueSize",
queue::size).register(registry);
}
508
Kotlin
import io.micrometer.core.instrument.Gauge
import io.micrometer.core.instrument.binder.MeterBinder
import org.springframework.context.annotation.Bean
class MyMeterBinderConfiguration {
@Bean
fun queueSize(queue: Queue): MeterBinder {
return MeterBinder { registry ->
Gauge.builder("queueSize", queue::size).register(registry)
}
}
Using a MeterBinder ensures that the correct dependency relationships are set up and that the bean
is available when the metric’s value is retrieved. A MeterBinder implementation can also be useful if
you find that you repeatedly instrument a suite of metrics across components or applications.
By default, metrics from all MeterBinder beans are automatically bound to the
NOTE
Spring-managed MeterRegistry.
If you need to apply customizations to specific Meter instances, you can use the
io.micrometer.core.instrument.config.MeterFilter interface.
For example, if you want to rename the mytag.region tag to mytag.area for all meter IDs beginning
with com.example, you can do the following:
Java
import io.micrometer.core.instrument.config.MeterFilter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyMetricsFilterConfiguration {
@Bean
public MeterFilter renameRegionTagMeterFilter() {
return MeterFilter.renameTag("com.example", "mytag.region", "mytag.area");
}
509
Kotlin
import io.micrometer.core.instrument.config.MeterFilter
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyMetricsFilterConfiguration {
@Bean
fun renameRegionTagMeterFilter(): MeterFilter {
return MeterFilter.renameTag("com.example", "mytag.region", "mytag.area")
}
Common Tags
Common tags are generally used for dimensional drill-down on the operating environment, such as
host, instance, region, stack, and others. Commons tags are applied to all meters and can be
configured, as the following example shows:
Properties
management.metrics.tags.region=us-east-1
management.metrics.tags.stack=prod
Yaml
management:
metrics:
tags:
region: "us-east-1"
stack: "prod"
The preceding example adds region and stack tags to all meters with a value of us-east-1 and prod,
respectively.
The order of common tags is important if you use Graphite. As the order of common
NOTE tags cannot be guaranteed by using this approach, Graphite users are advised to
define a custom MeterFilter instead.
510
Per-meter Properties
In addition to MeterFilter beans, you can apply a limited set of customization on a per-meter basis
using properties. Per-meter customizations are applied, using Spring Boot’s PropertiesMeterFilter,
to any meter IDs that start with the given name. The following example filters out any meters that
have an ID starting with example.remote.
Properties
management.metrics.enable.example.remote=false
Yaml
management:
metrics:
enable:
example:
remote: false
Property Description
management.metrics.enable Whether to accept meters with certain IDs.
Meters that are not accepted are filtered from
the MeterRegistry.
management.metrics.distribution.percentiles- Whether to publish a histogram suitable for
histogram computing aggregable (across dimension)
percentile approximations.
management.metrics.distribution.minimum- Publish fewer histogram buckets by clamping
expected-value, the range of expected values.
management.metrics.distribution.maximum-
expected-value
management.metrics.distribution.percentiles Publish percentile values computed in your
application
For more details on the concepts behind percentiles-histogram, percentiles, and slo, see the
“Histograms and percentiles” section of the Micrometer documentation.
511
13.7.6. Metrics Endpoint
Spring Boot provides a metrics endpoint that you can use diagnostically to examine the metrics
collected by an application. The endpoint is not available by default and must be exposed. See
exposing endpoints for more details.
Navigating to /actuator/metrics displays a list of available meter names. You can drill down to view
information about a particular meter by providing its name as a selector — for example,
/actuator/metrics/jvm.memory.max.
The name you use here should match the name used in the code, not the name after it
has been naming-convention normalized for a monitoring system to which it is
TIP shipped. In other words, if jvm.memory.max appears as jvm_memory_max in Prometheus
because of its snake case naming convention, you should still use jvm.memory.max as the
selector when inspecting the meter in the metrics endpoint.
You can also add any number of tag=KEY:VALUE query parameters to the end of the URL to
dimensionally drill down on a meter — for example,
/actuator/metrics/jvm.memory.max?tag=area:nonheap.
The reported measurements are the sum of the statistics of all meters that match the
meter name and any tags that have been applied. In the preceding example, the
returned Value statistic is the sum of the maximum memory footprints of the “Code
TIP Cache”, “Compressed Class Space”, and “Metaspace” areas of the heap. If you wanted
to see only the maximum size for the “Metaspace”, you could add an additional
tag=id:Metaspace — that is,
/actuator/metrics/jvm.memory.max?tag=area:nonheap&tag=id:Metaspace.
13.8. Tracing
Spring Boot Actuator provides dependency management and auto-configuration for Micrometer
Tracing, a facade for popular tracer libraries.
TIP To learn more about Micrometer Tracing capabilities, see its reference documentation.
512
13.8.2. Getting Started
We need an example application that we can use to get started with tracing. For our purposes, the
simple “Hello World!” web application that’s covered in the “Developing Your First Spring Boot
Application” section will suffice. We’re going to use the OpenTelemetry tracer with Zipkin as trace
backend.
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@SpringBootApplication
public class MyApplication {
@RequestMapping("/")
String home() {
logger.info("home() has been called");
return "Hello World!";
}
There’s an added logger statement in the home() method, which will be important
NOTE
later.
• org.springframework.boot:spring-boot-starter-actuator
513
Properties
management.tracing.sampling.probability=1.0
Yaml
management.tracing.sampling.probability: 1.0
By default, Spring Boot samples only 10% of requests to prevent overwhelming the trace backend.
This property switches it to 100% so that every request is sent to the trace backend.
To collect and visualize the traces, we need a running trace backend. We use Zipkin as our trace
backend here. The Zipkin Quickstart guide provides instructions how to start Zipkin locally.
If you open a web browser to localhost:8080, you should see the following output:
Hello World!
Behind the scenes, an observation has been created for the HTTP request, which in turn gets
bridged to OpenTelemetry, which reports a new trace to Zipkin.
Now open the Zipkin UI at localhost:9411 and press the "Run Query" button to list all collected
traces. You should see one trace. Press the "Show" button to see the details of that trace.
You can include the current trace and span id in the logs by setting the
TIP logging.pattern.level property to %5p [${spring.application.name:},%X{traceId:-
},%X{spanId:-}]
As Micrometer Tracer supports multiple tracer implementations, there are multiple dependency
combinations possible with Spring Boot.
Tracing with OpenTelemetry and reporting to Zipkin requires the following dependencies:
514
OpenTelemetry With Wavefront
Tracing with OpenTelemetry and reporting to Wavefront requires the following dependencies:
Tracing with OpenTelemetry and reporting using OTLP requires the following dependencies:
Tracing with OpenZipkin Brave and reporting to Zipkin requires the following dependencies:
Tracing with OpenZipkin Brave and reporting to Wavefront requires the following dependencies:
515
13.8.5. Creating Custom Spans
You can create your own spans by starting an observation. For this, inject ObservationRegistry into
your component:
import io.micrometer.observation.Observation;
import io.micrometer.observation.ObservationRegistry;
import org.springframework.stereotype.Component;
@Component
class CustomObservation {
CustomObservation(ObservationRegistry observationRegistry) {
this.observationRegistry = observationRegistry;
}
void someOperation() {
Observation observation = Observation.createNotStarted("some-operation",
this.observationRegistry);
observation.lowCardinalityKeyValue("some-tag", "some-value");
observation.observe(() -> {
// Business logic ...
});
}
This will create an observation named "some-operation" with the tag "some-tag=some-value".
If you want to create a span without creating a metric, you need to use the lower-level
TIP
Tracer API from Micrometer.
13.9. Auditing
Once Spring Security is in play, Spring Boot Actuator has a flexible audit framework that publishes
events (by default, “authentication success”, “failure” and “access denied” exceptions). This feature
can be very useful for reporting and for implementing a lock-out policy based on authentication
failures.
You can enable auditing by providing a bean of type AuditEventRepository in your application’s
configuration. For convenience, Spring Boot offers an InMemoryAuditEventRepository.
InMemoryAuditEventRepository has limited capabilities, and we recommend using it only for
development environments. For production environments, consider creating your own alternative
AuditEventRepository implementation.
516
13.9.1. Custom Auditing
To customize published security events, you can provide your own implementations of
AbstractAuthenticationAuditListener and AbstractAuthorizationAuditListener.
You can also use the audit services for your own business events. To do so, either inject the
AuditEventRepository bean into your own components and use that directly or publish an
AuditApplicationEvent with the Spring ApplicationEventPublisher (by implementing
ApplicationEventPublisherAware).
You can use the httpexchanges endpoint to obtain information about the request-response
exchanges that are stored in the HttpExchangeRepository.
To customize the items that are included in each recorded exchange, use the
management.httpexchanges.recording.include configuration property.
• ApplicationPidFileWriter creates a file that contains the application PID (by default, in the
application directory with a file name of application.pid).
• WebServerPortFileWriter creates a file (or files) that contain the ports of the running web server
(by default, in the application directory with a file name of application.port).
By default, these writers are not activated, but you can enable them:
• By Extending Configuration
In the META-INF/spring.factories file, you can activate the listener (or listeners) that writes a PID
517
file:
org.springframework.context.ApplicationListener=\
org.springframework.boot.context.ApplicationPidFileWriter,\
org.springframework.boot.web.context.WebServerPortFileWriter
You can also activate a listener by invoking the SpringApplication.addListeners(…) method and
passing the appropriate Writer object. This method also lets you customize the file name and path
in the Writer constructor.
The extended support lets Cloud Foundry management UIs (such as the web application that you
can use to view deployed applications) be augmented with Spring Boot actuator information. For
example, an application status page can include full health information instead of the typical
“running” or “stopped” status.
If you want to fully disable the /cloudfoundryapplication endpoints, you can add the following
setting to your application.properties file:
Properties
management.cloudfoundry.enabled=false
Yaml
management:
cloudfoundry:
enabled: false
By default, the security verification for /cloudfoundryapplication endpoints makes SSL calls to
various Cloud Foundry services. If your Cloud Foundry UAA or Cloud Controller services use self-
signed certificates, you need to set the following property:
518
Properties
management.cloudfoundry.skip-ssl-validation=true
Yaml
management:
cloudfoundry:
skip-ssl-validation: true
If the server’s context-path has been configured to anything other than /, the Cloud Foundry
endpoints are not available at the root of the application. For example, if server.servlet.context-
path=/app, Cloud Foundry endpoints are available at /app/cloudfoundryapplication/*.
Java
import java.io.IOException;
import java.util.Collections;
import jakarta.servlet.GenericServlet;
import jakarta.servlet.Servlet;
import jakarta.servlet.ServletContainerInitializer;
import jakarta.servlet.ServletContext;
import jakarta.servlet.ServletException;
import jakarta.servlet.ServletRequest;
import jakarta.servlet.ServletResponse;
import org.apache.catalina.Host;
import org.apache.catalina.core.StandardContext;
import org.apache.catalina.startup.Tomcat;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.servlet.ServletContextInitializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyCloudFoundryConfiguration {
@Bean
public TomcatServletWebServerFactory servletWebServerFactory() {
return new TomcatServletWebServerFactory() {
519
@Override
protected void prepareContext(Host host, ServletContextInitializer[]
initializers) {
super.prepareContext(host, initializers);
StandardContext child = new StandardContext();
child.addLifecycleListener(new Tomcat.FixContextListener());
child.setPath("/cloudfoundryapplication");
ServletContainerInitializer initializer =
getServletContextInitializer(getContextPath());
child.addServletContainerInitializer(initializer,
Collections.emptySet());
child.setCrossContext(true);
host.addChild(child);
}
};
}
@Override
public void service(ServletRequest req, ServletResponse res) throws
ServletException, IOException {
ServletContext context =
req.getServletContext().getContext(contextPath);
context.getRequestDispatcher("/cloudfoundryapplication").forward(req, res);
}
};
context.addServlet("cloudfoundry", servlet).addMapping("/*");
};
}
Kotlin
import jakarta.servlet.GenericServlet
import jakarta.servlet.Servlet
import jakarta.servlet.ServletContainerInitializer
import jakarta.servlet.ServletContext
import jakarta.servlet.ServletException
import jakarta.servlet.ServletRequest
import jakarta.servlet.ServletResponse
import org.apache.catalina.Host
import org.apache.catalina.core.StandardContext
import org.apache.catalina.startup.Tomcat.FixContextListener
520
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory
import org.springframework.boot.web.servlet.ServletContextInitializer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import java.io.IOException
import java.util.Collections.emptySet
@Configuration(proxyBeanMethods = false)
class MyCloudFoundryConfiguration {
@Bean
fun servletWebServerFactory(): TomcatServletWebServerFactory {
return object : TomcatServletWebServerFactory() {
}
}
@Throws(ServletException::class, IOException::class)
override fun service(req: ServletRequest, res: ServletResponse) {
val servletContext = req.servletContext.getContext(contextPath)
servletContext.getRequestDispatcher("/cloudfoundryapplication").forward(req, res)
}
}
context.addServlet("cloudfoundry", servlet).addMapping("/*")
}
}
}
521
13.13. What to Read Next
You might want to read about graphing tools such as Graphite.
Otherwise, you can continue on to read about “deployment options” or jump ahead for some in-
depth information about Spring Boot’s build tool plugins.
522
Chapter 14. Deploying Spring Boot
Applications
Spring Boot’s flexible packaging options provide a great deal of choice when it comes to deploying
your application. You can deploy Spring Boot applications to a variety of cloud platforms, to
virtual/real machines, or make them fully executable for Unix systems.
Two popular cloud providers, Heroku and Cloud Foundry, employ a “buildpack” approach. The
buildpack wraps your deployed code in whatever is needed to start your application. It might be a
JDK and a call to java, an embedded web server, or a full-fledged application server. A buildpack is
pluggable, but ideally you should be able to get by with as few customizations to it as possible. This
reduces the footprint of functionality that is not under your control. It minimizes divergence
between development and production environments.
Ideally, your application, like a Spring Boot executable jar, has everything that it needs to run
packaged within it.
In this section, we look at what it takes to get the application that we developed in the “Getting
Started” section up and running in the Cloud.
Cloud Foundry provides default buildpacks that come into play if no other buildpack is specified.
The Cloud Foundry Java buildpack has excellent support for Spring applications, including Spring
Boot. You can deploy stand-alone executable jar applications as well as traditional .war packaged
applications.
Once you have built your application (by using, for example, mvn clean package) and have installed
the cf command line tool, deploy your application by using the cf push command, substituting the
path to your compiled .jar. Be sure to have logged in with your cf command line client before
pushing an application. The following line shows using the cf push command to deploy an
application:
523
See the cf push documentation for more options. If there is a Cloud Foundry manifest.yml file
present in the same directory, it is considered.
At this point, cf starts uploading your application, producing output similar to the following
example:
Uploading acloudyspringtime... OK
Preparing to start acloudyspringtime... OK
-----> Downloaded app package (8.9M)
-----> Java Buildpack Version: v3.12 (offline) | https://github.com/cloudfoundry/java-
buildpack.git#6f25b7e
-----> Downloading Open Jdk JRE
Expanding Open Jdk JRE to .java-buildpack/open_jdk_jre (1.6s)
-----> Downloading Open JDK Like Memory Calculator 2.0.2_RELEASE from https://java-
buildpack.cloudfoundry.org/memory-calculator/trusty/x86_64/memory-calculator-
2.0.2_RELEASE.tar.gz (found in cache)
Memory Settings: -Xss349K -Xmx681574K -XX:MaxMetaspaceSize=104857K -Xms681574K
-XX:MetaspaceSize=104857K
-----> Downloading Container Certificate Trust Store 1.0.0_RELEASE from https://java-
buildpack.cloudfoundry.org/container-certificate-trust-store/container-certificate-
trust-store-1.0.0_RELEASE.jar (found in cache)
Adding certificates to .java-
buildpack/container_certificate_trust_store/truststore.jks (0.6s)
-----> Downloading Spring Auto Reconfiguration 1.10.0_RELEASE from https://java-
buildpack.cloudfoundry.org/auto-reconfiguration/auto-reconfiguration-
1.10.0_RELEASE.jar (found in cache)
Checking status of app 'acloudyspringtime'...
0 of 1 instances running (1 starting)
...
0 of 1 instances running (1 starting)
...
0 of 1 instances running (1 starting)
...
1 of 1 instances running (1 running)
App started
Once your application is live, you can verify the status of the deployed application by using the cf
apps command, as shown in the following example:
524
$ cf apps
Getting applications in ...
OK
Once Cloud Foundry acknowledges that your application has been deployed, you should be able to
find the application at the URI given. In the preceding example, you could find it at
https://acloudyspringtime.cfapps.io/.
Binding to Services
By default, metadata about the running application as well as service connection information is
exposed to the application as environment variables (for example: $VCAP_SERVICES). This
architecture decision is due to Cloud Foundry’s polyglot (any language and platform can be
supported as a buildpack) nature. Process-scoped environment variables are language agnostic.
Environment variables do not always make for the easiest API, so Spring Boot automatically
extracts them and flattens the data into properties that can be accessed through Spring’s
Environment abstraction, as shown in the following example:
Java
import org.springframework.context.EnvironmentAware;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Component;
@Component
public class MyBean implements EnvironmentAware {
@Override
public void setEnvironment(Environment environment) {
this.instanceId = environment.getProperty("vcap.application.instance_id");
}
// ...
525
Kotlin
import org.springframework.context.EnvironmentAware
import org.springframework.core.env.Environment
import org.springframework.stereotype.Component
@Component
class MyBean : EnvironmentAware {
// ...
All Cloud Foundry properties are prefixed with vcap. You can use vcap properties to access
application information (such as the public URL of the application) and service information (such
as database credentials). See the CloudFoundryVcapEnvironmentPostProcessor Javadoc for complete
details.
TIP The Java CFEnv project is a better fit for tasks such as configuring a DataSource.
14.1.2. Kubernetes
Spring Boot auto-detects Kubernetes deployment environments by checking the environment for
"*_SERVICE_HOST" and "*_SERVICE_PORT" variables. You can override this detection with the
spring.main.cloud-platform configuration property.
Spring Boot helps you to manage the state of your application and export it with HTTP Kubernetes
Probes using Actuator.
When Kubernetes deletes an application instance, the shutdown process involves several
subsystems concurrently: shutdown hooks, unregistering the service, removing the instance from
the load-balancer… Because this shutdown processing happens in parallel (and due to the nature of
distributed systems), there is a window during which traffic can be routed to a pod that has also
begun its shutdown processing.
You can configure a sleep execution in a preStop handler to avoid requests being routed to a pod
that has already begun shutting down. This sleep should be long enough for new requests to stop
being routed to the pod and its duration will vary from deployment to deployment. The preStop
handler can be configured by using the PodSpec in the pod’s configuration file as follows:
526
spec:
containers:
- name: "example-container"
image: "example-image"
lifecycle:
preStop:
exec:
command: ["sh", "-c", "sleep 10"]
Once the pre-stop hook has completed, SIGTERM will be sent to the container and graceful
shutdown will begin, allowing any remaining in-flight requests to complete.
When Kubernetes sends a SIGTERM signal to the pod, it waits for a specified time
called the termination grace period (the default for which is 30 seconds). If the
containers are still running after the grace period, they are sent the SIGKILL signal
NOTE and forcibly removed. If the pod takes longer than 30 seconds to shut down, which
could be because you have increased spring.lifecycle.timeout-per-shutdown-phase,
make sure to increase the termination grace period by setting the
terminationGracePeriodSeconds option in the Pod YAML.
14.1.3. Heroku
Heroku is another popular PaaS platform. To customize Heroku builds, you provide a Procfile,
which provides the incantation required to deploy an application. Heroku assigns a port for the
Java application to use and then ensures that routing to the external URI works.
You must configure your application to listen on the correct port. The following example shows the
Procfile for our starter REST application:
Spring Boot makes -D arguments available as properties accessible from a Spring Environment
instance. The server.port configuration property is fed to the embedded Tomcat, Jetty, or Undertow
instance, which then uses the port when it starts up. The $PORT environment variable is assigned to
us by the Heroku PaaS.
This should be everything you need. The most common deployment workflow for Heroku
deployments is to git push the code to production, as shown in the following example:
527
Initializing repository, done.
Counting objects: 95, done.
Delta compression using up to 8 threads.
Compressing objects: 100% (78/78), done.
Writing objects: 100% (95/95), 8.66 MiB | 606.00 KiB/s, done.
Total 95 (delta 31), reused 0 (delta 0)
To git@heroku.com:agile-sierra-1405.git
* [new branch] main -> main
Your application should now be up and running on Heroku. For more details, see Deploying Spring
Boot Applications to Heroku.
14.1.4. OpenShift
OpenShift has many resources describing how to deploy Spring Boot applications, including:
• Architecture guide
528
• OpenShift Commons Briefing
Amazon Web Services offers multiple ways to install Spring Boot-based applications, either as
traditional web applications (war) or as executable jar files with an embedded web server. The
options include:
Each has different features and pricing models. In this document, we describe to approach using
AWS Elastic Beanstalk.
As described in the official Elastic Beanstalk Java guide, there are two main options to deploy a Java
application. You can either use the “Tomcat Platform” or the “Java SE platform”.
This option applies to Spring Boot projects that produce a war file. No special configuration is
required. You need only follow the official guide.
This option applies to Spring Boot projects that produce a jar file and run an embedded web
container. Elastic Beanstalk environments run an nginx instance on port 80 to proxy the actual
application, running on port 5000. To configure it, add the following line to your
application.properties file:
server.port=5000
deploy:
artifact: target/demo-0.0.1-SNAPSHOT.jar
529
Reduce costs by setting the environment type
By default an Elastic Beanstalk environment is load balanced. The load balancer has a
significant cost. To avoid that cost, set the environment type to “Single instance”, as
described in the Amazon documentation. You can also create single instance
TIP
environments by using the CLI and the following command:
eb create -s
Summary
This is one of the easiest ways to get to AWS, but there are more things to cover, such as how to
integrate Elastic Beanstalk into any CI / CD tool, use the Elastic Beanstalk Maven plugin instead of
the CLI, and others. There is a blog post covering these topics more in detail.
CloudCaptain works by turning your Spring Boot executable jar or war into a minimal VM image
that can be deployed unchanged either on VirtualBox or on AWS. CloudCaptain comes with deep
integration for Spring Boot and uses the information from your Spring Boot configuration file to
automatically configure ports and health check URLs. CloudCaptain leverages this information both
for the images it produces as well as for all the resources it provisions (instances, security groups,
elastic load balancers, and so on).
Once you have created a CloudCaptain account, connected it to your AWS account, installed the
latest version of the CloudCaptain Client, and ensured that the application has been built by Maven
or Gradle (by using, for example, mvn clean package), you can deploy your Spring Boot application
to AWS with a command similar to the following:
See the boxfuse run documentation for more options. If there is a boxfuse.conf file present in the
current directory, it is considered.
At this point, CloudCaptain creates an image for your application, uploads it, and configures and
starts the necessary resources on AWS, resulting in output similar to the following example:
530
Fusing Image for myapp-1.0.jar ...
Image fused in 00:06.838s (53937 K) -> axelfontaine/myapp:1.0
Creating axelfontaine/myapp ...
Pushing axelfontaine/myapp:1.0 ...
Verifying axelfontaine/myapp:1.0 ...
Creating Elastic IP ...
Mapping myapp-axelfontaine.boxfuse.io to 52.28.233.167 ...
Waiting for AWS to create an AMI for axelfontaine/myapp:1.0 in eu-central-1 (this may
take up to 50 seconds) ...
AMI created in 00:23.557s -> ami-d23f38cf
Creating security group boxfuse-sg_axelfontaine/myapp:1.0 ...
Launching t2.micro instance of axelfontaine/myapp:1.0 (ami-d23f38cf) in eu-central-1
...
Instance launched in 00:30.306s -> i-92ef9f53
Waiting for AWS to boot Instance i-92ef9f53 and Payload to start at
https://52.28.235.61/ ...
Payload started in 00:29.266s -> https://52.28.235.61/
Remapping Elastic IP 52.28.233.167 to i-92ef9f53 ...
Waiting 15s for AWS to complete Elastic IP Zero Downtime transition ...
Deployment completed successfully. axelfontaine/myapp:1.0 is up and running at
https://myapp-axelfontaine.boxfuse.io/
See the blog post on deploying Spring Boot apps on EC2 as well as the documentation for the
CloudCaptain Spring Boot integration to get started with a Maven build to run the app.
14.1.7. Azure
This Getting Started guide walks you through deploying your Spring Boot application to either
Azure Spring Cloud or Azure App Service.
Google Cloud has several options that can be used to launch Spring Boot applications. The easiest to
get started with is probably App Engine, but you could also find ways to run Spring Boot in a
container with Container Engine or on a virtual machine with Compute Engine.
To run in App Engine, you can create a project in the UI first, which sets up a unique identifier for
you and also sets up HTTP routes. Add a Java app to the project and leave it empty and then use the
Google Cloud SDK to push your Spring Boot app into that slot from the command line or CI build.
App Engine Standard requires you to use WAR packaging. Follow these steps to deploy App Engine
Standard application to Google Cloud.
Alternatively, App Engine Flex requires you to create an app.yaml file to describe the resources your
app requires. Normally, you put this file in src/main/appengine, and it should resemble the following
file:
531
service: "default"
runtime: "java"
env: "flex"
runtime_config:
jdk: "openjdk8"
handlers:
- url: "/.*"
script: "this field is required, but ignored"
manual_scaling:
instances: 1
health_check:
enable_health_check: false
env_variables:
ENCRYPT_KEY: "your_encryption_key_here"
You can deploy the app (for example, with a Maven plugin) by adding the project ID to the build
configuration, as shown in the following example:
<plugin>
<groupId>com.google.cloud.tools</groupId>
<artifactId>appengine-maven-plugin</artifactId>
<version>1.3.0</version>
<configuration>
<project>myproject</project>
</configuration>
</plugin>
Then deploy with mvn appengine:deploy (you need to authenticate first, otherwise the build fails).
532
Fully executable jars work by embedding an extra script at the front of the file.
Currently, some tools do not accept this format, so you may not always be able
to use this technique. For example, jar -xf may silently fail to extract a jar or
CAUTION
war that has been made fully executable. It is recommended that you make
your jar or war fully executable only if you intend to execute it directly, rather
than running it with java -jar or deploying it to a servlet container.
To create a ‘fully executable’ jar with Maven, use the following plugin configuration:
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<executable>true</executable>
</configuration>
</plugin>
tasks.named('bootJar') {
launchScript()
}
You can then run your application by typing ./my-application.jar (where my-application is the
name of your artifact). The directory containing the jar is used as your application’s working
directory.
The default script supports most Linux distributions and is tested on CentOS and Ubuntu. Other
platforms, such as OS X and FreeBSD, require the use of a custom embeddedLaunchScript.
Spring Boot application can be easily started as Unix/Linux services by using either init.d or
systemd.
If you configured Spring Boot’s Maven or Gradle plugin to generate a fully executable jar, and you
do not use a custom embeddedLaunchScript, your application can be used as an init.d service. To do
533
so, symlink the jar to init.d to support the standard start, stop, restart, and status commands.
• Starts the services as the user that owns the jar file
Assuming that you have a Spring Boot application installed in /var/myapp, to install a Spring Boot
application as an init.d service, create a symlink, as follows:
Once installed, you can start and stop the service in the usual way. For example, on a Debian-based
system, you could start it with the following command:
If your application fails to start, check the log file written to /var/log/<appname>.log for
TIP
errors.
You can also flag the application to start automatically by using your standard operating system
tools. For example, on Debian, you could use the following command:
The following is a set of guidelines on how to secure a Spring Boot application that
NOTE runs as an init.d service. It is not intended to be an exhaustive list of everything that
should be done to harden an application and the environment in which it runs.
When executed as root, as is the case when root is being used to start an init.d service, the default
executable script runs the application as the user specified in the RUN_AS_USER environment
variable. When the environment variable is not set, the user who owns the jar file is used instead.
You should never run a Spring Boot application as root, so RUN_AS_USER should never be root and
your application’s jar file should never be owned by root. Instead, create a specific user to run your
application and set the RUN_AS_USER environment variable or use chown to make it the owner of the
jar file, as shown in the following example:
In this case, the default executable script runs the application as the bootapp user.
534
To reduce the chances of the application’s user account being compromised, you
TIP should consider preventing it from using a login shell. For example, you can set the
account’s shell to /usr/sbin/nologin.
You should also take steps to prevent the modification of your application’s jar file. Firstly,
configure its permissions so that it cannot be written and can only be read or executed by its
owner, as shown in the following example:
Second, you should also take steps to limit the damage if your application or the account that is
running it is compromised. If an attacker does gain access, they could make the jar file writable and
change its contents. One way to protect against this is to make it immutable by using chattr, as
shown in the following example:
This will prevent any user, including root, from modifying the jar.
If root is used to control the application’s service and you use a .conf file to customize its startup,
the .conf file is read and evaluated by the root user. It should be secured accordingly. Use chmod so
that the file can only be read by the owner and use chown to make root the owner, as shown in the
following example:
systemd is the successor of the System V init system and is now being used by many modern Linux
distributions. Although you can continue to use init.d scripts with systemd, it is also possible to
launch Spring Boot applications by using systemd ‘service’ scripts.
Assuming that you have a Spring Boot application installed in /var/myapp, to install a Spring Boot
application as a systemd service, create a script named myapp.service and place it in
/etc/systemd/system directory. The following script offers an example:
535
[Unit]
Description=myapp
After=syslog.target
[Service]
User=myapp
ExecStart=/var/myapp/myapp.jar
SuccessExitStatus=143
[Install]
WantedBy=multi-user.target
Remember to change the Description, User, and ExecStart fields for your
IMPORTANT
application.
The ExecStart field does not declare the script action command, which means that
NOTE
the run command is used by default.
Note that, unlike when running as an init.d service, the user that runs the application, the PID file,
and the console log file are managed by systemd itself and therefore must be configured by using
appropriate fields in the ‘service’ script. Consult the service unit configuration man page for more
details.
To flag the application to start automatically on system boot, use the following command:
The default embedded startup script written by the Maven or Gradle plugin can be customized in a
number of ways. For most people, using the default script along with a few customizations is
usually enough. If you find you cannot customize something that you need to, use the
embeddedLaunchScript option to write your own file entirely.
It often makes sense to customize elements of the start script as it is written into the jar file. For
example, init.d scripts can provide a “description”. Since you know the description up front (and it
need not change), you may as well provide it when the jar is generated.
To customize written elements, use the embeddedLaunchScriptProperties option of the Spring Boot
Maven plugin or the properties property of the Spring Boot Gradle plugin’s launchScript.
The following property substitutions are supported with the default script:
536
Name Description Gradle default Maven default
mode The script mode. auto auto
537
Name Description Gradle default Maven default
pidFilen Default value for the name
ame of the PID file in PID_FOLDER.
Only valid for an init.d
service
useStart Whether the start-stop- true true
StopDaem daemon command, when it is
on
available, should be used to
control the process
stopWait Default value for 60 60
Time STOP_WAIT_TIME in seconds.
Only valid for an init.d
service
For items of the script that need to be customized after the jar has been written, you can use
environment variables or a config file.
The following environment properties are supported with the default script:
Variable Description
MODE The “mode” of operation. The default depends on the way the jar was built but is
usually auto (meaning it tries to guess if it is an init script by checking if it is a
symlink in a directory called init.d). You can explicitly set it to service so that the
stop|start|status|restart commands work or to run if you want to run the script in
the foreground.
RUN_AS_USER The user that will be used to run the application. When not set, the user that owns
the jar file will be used.
USE_START_ST Whether the start-stop-daemon command, when it is available, should be used to
OP_DAEMON control the process. Defaults to true.
PID_FOLDER The root name of the pid folder (/var/run by default).
LOG_FOLDER The name of the folder in which to put log files (/var/log by default).
CONF_FOLDER The name of the folder from which to read .conf files (same folder as jar-file by
default).
LOG_FILENAME The name of the log file in the LOG_FOLDER (<appname>.log by default).
APP_NAME The name of the app. If the jar is run from a symlink, the script guesses the app
name. If it is not a symlink or you want to explicitly set the app name, this can be
useful.
RUN_ARGS The arguments to pass to the program (the Spring Boot app).
JAVA_HOME The location of the java executable is discovered by using the PATH by default, but
you can set it explicitly if there is an executable file at $JAVA_HOME/bin/java.
538
Variable Description
JAVA_OPTS Options that are passed to the JVM when it is launched.
JARFILE The explicit location of the jar file, in case the script is being used to launch a jar
that it is not actually embedded.
DEBUG If not empty, sets the -x flag on the shell process, allowing you to see the logic in the
script.
STOP_WAIT_TI The time in seconds to wait when stopping the application before forcing a
ME shutdown (60 by default).
The PID_FOLDER, LOG_FOLDER, and LOG_FILENAME variables are only valid for an init.d
NOTE service. For systemd, the equivalent customizations are made by using the ‘service’
script. See the service unit configuration man page for more details.
With the exception of JARFILE and APP_NAME, the settings listed in the preceding section can be
configured by using a .conf file. The file is expected to be next to the jar file and have the same
name but suffixed with .conf rather than .jar. For example, a jar named /var/myapp/myapp.jar uses
the configuration file named /var/myapp/myapp.conf, as shown in the following example:
myapp.conf
JAVA_OPTS=-Xmx1024M
LOG_FOLDER=/custom/log/folder
If you do not like having the config file next to the jar file, you can set a CONF_FOLDER
TIP
environment variable to customize the location of the config file.
To learn about securing this file appropriately, see the guidelines for securing an init.d service.
A (separately maintained sample) describes step-by-step how you can create a Windows service for
your Spring Boot application.
If you are running your application from a container, you can use an executable jar, but it is also
often an advantage to explode it and run it in a different way. Certain PaaS implementations may
also choose to unpack archives before they run. For example, Cloud Foundry operates this way. One
way to run an unpacked archive is by starting the appropriate launcher, as follows:
539
$ jar -xf myapp.jar
$ java org.springframework.boot.loader.JarLauncher
This is actually slightly faster on startup (depending on the size of the jar) than running from an
unexploded archive. After startup, you should not expect any differences.
Once you have unpacked the jar file, you can also get an extra boost to startup time by running the
app with its "natural" main method instead of the JarLauncher. For example:
Using the JarLauncher over the application’s main method has the added benefit of a
NOTE predictable classpath order. The jar contains a classpath.idx file which is used by
the JarLauncher when constructing the classpath.
It’s beneficial for the startup time to run your application using the AOT generated initialization
code. First, you need to ensure that the jar you are building includes AOT generated code.
For Maven, this means that you should build with -Pnative to activate the native profile:
For Gradle, you need to ensure that your build includes the org.springframework.boot.aot plugin.
When the JAR has been built, run it with spring.aot.enabled system property set to true. For
example:
Beware that using the ahead-of-time processing has drawbacks. It implies the following restrictions:
◦ Properties that change if a bean is created are not supported (for example,
@ConditionalOnProperty and .enable properties).
To learn more about ahead-of-time processing, please see the Understanding Spring Ahead-of-Time
Processing section.
540
14.4. What to Read Next
See the Cloud Foundry, Heroku, OpenShift, and Boxfuse web sites for more information about the
kinds of features that a PaaS can offer. These are just four of the most popular Java PaaS providers.
Since Spring Boot is so amenable to cloud-based deployment, you can freely consider other
providers as well.
The next section goes on to cover the GraalVM Native Images, or you can jump ahead to read about
the Spring Boot CLI or our build tool plugins.
541
Chapter 15. GraalVM Native Image Support
GraalVM Native Images are standalone executables that can be generated by processing compiled
Java applications ahead-of-time. Native Images generally have a smaller memory footprint and
start faster than their JVM counterparts.
They are well suited to applications that are deployed using container images and are especially
interesting when combined with "Function as a service" (FaaS) platforms.
Unlike traditional applications written for the JVM, GraalVM Native Image applications require
ahead-of-time processing in order to create an executable. This ahead-of-time processing involves
statically analyzing your application code from its main entry point.
A GraalVM Native Image is a complete, platform-specific executable. You do not need to ship a Java
Virtual Machine in order to run a native image.
If you just want to get started and experiment with GraalVM you can skip ahead to the
TIP “Developing Your First GraalVM Native Application” section and return to this section
later.
The fact that GraalVM Native Images are produced ahead-of-time means that there are some key
differences between native and JVM based applications. The main differences are:
• Static analysis of your application is performed at build-time from the main entry point.
• Code that cannot be reached when the native image is created will be removed and won’t be
part of the executable.
• GraalVM is not directly aware of dynamic elements of your code and must be told about
reflection, resources, serialization, and dynamic proxies.
• There is no lazy class loading, everything shipped in the executables will be loaded in memory
on startup.
• There are some limitations around some aspects of Java applications that are not fully
supported.
542
15.1.2. Understanding Spring Ahead-of-Time Processing
Typical Spring Boot applications are quite dynamic and configuration is performed at runtime. In
fact, the concept of Spring Boot auto-configuration depends heavily on reacting to the state of the
runtime in order to configure things correctly.
Although it would be possible to tell GraalVM about these dynamic aspects of the application, doing
so would undo most of the benefit of static analysis. So instead, when using Spring Boot to create
native images, a closed-world is assumed and the dynamic aspects of the application are restricted.
◦ Properties that change if a bean is created are not supported (for example,
@ConditionalOnProperty and .enable properties).
When these restrictions are in place, it becomes possible for Spring to perform ahead-of-time
processing during build-time and generate additional assets that GraalVM can use. A Spring AOT
processed application will typically generate:
Spring applications are composed of Spring Beans. Internally, Spring Framework uses two distinct
concepts to manage beans. There are bean instances, which are the actual instances that have been
created and can be injected into other beans. There are also bean definitions which are used to
define attributes of a bean and how its instance should be created.
543
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyConfiguration {
@Bean
public MyBean myBean() {
return new MyBean();
}
The bean definition is created by parsing the @Configuration class and finding the @Bean methods. In
the above example, we’re defining a BeanDefinition for a singleton bean named myBean. We’re also
creating a BeanDefinition for the MyConfiguration class itself.
When the myBean instance is required, Spring knows that it must invoke the myBean() method and
use the result. When running on the JVM, @Configuration class parsing happens when your
application starts and @Bean methods are invoked using reflection.
When creating a native image, Spring operates in a different way. Rather than parsing
@Configuration classes and generating bean definitions at runtime, it does it at build-time. Once the
bean definitions have been discovered, they are processed and converted into source code that can
be analyzed by the GraalVM compiler.
The Spring AOT process would convert the configuration class above to code like this:
544
import org.springframework.beans.factory.aot.BeanInstanceSupplier;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.RootBeanDefinition;
/**
* Bean definitions for {@link MyConfiguration}.
*/
public class MyConfiguration__BeanDefinitions {
/**
* Get the bean definition for 'myConfiguration'.
*/
public static BeanDefinition getMyConfigurationBeanDefinition() {
Class<?> beanType = MyConfiguration.class;
RootBeanDefinition beanDefinition = new RootBeanDefinition(beanType);
beanDefinition.setInstanceSupplier(MyConfiguration::new);
return beanDefinition;
}
/**
* Get the bean instance supplier for 'myBean'.
*/
private static BeanInstanceSupplier<MyBean> getMyBeanInstanceSupplier() {
return BeanInstanceSupplier.<MyBean>forFactoryMethod(MyConfiguration.class,
"myBean")
.withGenerator((registeredBean) ->
registeredBean.getBeanFactory().getBean(MyConfiguration.class).myBean());
}
/**
* Get the bean definition for 'myBean'.
*/
public static BeanDefinition getMyBeanBeanDefinition() {
Class<?> beanType = MyBean.class;
RootBeanDefinition beanDefinition = new RootBeanDefinition(beanType);
beanDefinition.setInstanceSupplier(getMyBeanInstanceSupplier());
return beanDefinition;
}
The exact code generated may differ depending on the nature of your bean
NOTE
definitions.
You can see above that the generated code creates equivalent bean definitions to the @Configuration
class, but in a direct way that can be understood by GraalVM.
There is a bean definition for the myConfiguration bean, and one for myBean. When a myBean instance
is required, a BeanInstanceSupplier is called. This supplier will invoke the myBean() method on the
545
myConfiguration bean.
During Spring AOT processing your application is started up to the point that bean
NOTE definitions are available. Bean instances are not created during the AOT processing
phase.
Spring AOT will generate code like this for all your bean definitions. It will also generate code when
bean post-processing is required (for example, to call @Autowired methods). An
ApplicationContextInitializer will also be generated which will be used by Spring Boot to initialize
the ApplicationContext when an AOT processed application is actually run.
Although AOT generated source code can be verbose, it is quite readable and can be
helpful when debugging an application. Generated source files can be found in
TIP
target/spring-aot/main/sources when using Maven and build/generated/aotSources
with Gradle.
In addition to generating source files, the Spring AOT engine will also generate hint files that are
used by GraalVM. Hint files contain JSON data that describes how GraalVM should deal with things
that it can’t understand by directly inspecting the code.
For example, you might be using a Spring annotation on a private method. Spring will need to use
reflection in order to invoke private methods, even on GraalVM. When such situations arise, Spring
can write a reflection hint so that GraalVM knows that even though the private method isn’t called
directly, it still needs to be available in the native image.
Hint files are generated under META-INF/native-image where they are automatically picked up by
GraalVM.
Spring sometimes needs to generate proxy classes to enhance the code you’ve written with
additional features. To do this, it uses the cglib library which directly generates bytecode.
When an application is running on the JVM, proxy classes are generated dynamically as the
application runs. When creating a native image, these proxies need to be created at build-time so
that they can be included by GraalVM.
Unlike source code generation, generated bytecode isn’t particularly helpful when
debugging an application. However, if you need to inspect the contents of the .class
NOTE
files using a tool such as javap you can find them in target/spring-aot/main/classes
for Maven and build/generated/aotClasses for Gradle.
546
15.2. Developing Your First GraalVM Native
Application
Now that we have a good overview of GraalVM Native Images and how the Spring ahead-of-time
engine works, we can look at how to create an application.
There are two main ways to build a Spring Boot native image application:
• Using Spring Boot support for Cloud Native Buildpacks to generate a lightweight container
containing a native executable.
The easiest way to start a new native Spring Boot project is to go to start.spring.io, add
TIP the “GraalVM Native Support” dependency and generate the project. The included
HELP.md file will provide getting started hints.
We need an example application that we can use to create our native image. For our purposes, the
simple “Hello World!” web application that’s covered in the “Developing Your First Spring Boot
Application” section will suffice.
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@SpringBootApplication
public class MyApplication {
@RequestMapping("/")
String home() {
return "Hello World!";
}
This application uses Spring MVC and embedded Tomcat, both of which have been tested and
verified to work with GraalVM native images.
547
15.2.2. Building a Native Image Using Buildpacks
Spring Boot includes buildpack support for native images directly for both Maven and Gradle. This
means you can just type a single command and quickly get a sensible image into your locally
running Docker daemon. The resulting image doesn’t contain a JVM, instead the native image is
compiled statically. This leads to smaller images.
System Requirements
Docker should be installed. See Get Docker for more details. Configure it to allow non-root user if
you are on Linux.
You can run docker run hello-world (without sudo) to check the Docker daemon is
NOTE reachable as expected. Check the Maven or Gradle Spring Boot plugin
documentation for more details.
Using Maven
To build a native image container using Maven you should ensure that your pom.xml file uses the
spring-boot-starter-parent and the org.graalvm.buildtools:native-maven-plugin. You should have a
<parent> section that looks like this:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>3.1.0-RC2</version>
</parent>
<plugin>
<groupId>org.graalvm.buildtools</groupId>
<artifactId>native-maven-plugin</artifactId>
</plugin>
The spring-boot-starter-parent declares a native profile that configures the executions that need to
548
run in order to create a native image. You can activate profiles using the -P flag on the command
line.
To build the image, you can run the spring-boot:build-image goal with the native profile active:
Using Gradle
The Spring Boot Gradle plugin automatically configures AOT tasks when the GraalVM Native Image
plugin is applied. You should check that your Gradle build contains a plugins block that includes
org.graalvm.buildtools.native.
As long as the org.graalvm.buildtools.native plugin is applied, the bootBuildImage task will generate
a native image rather than a JVM one. You can run the task using:
$ gradle bootBuildImage
Once you have run the appropriate build command, a Docker image should be available. You can
start your application using docker run:
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.08 seconds (process running for 0.095)
The startup time differs from machine to machine, but it should be much faster
NOTE
than a Spring Boot application running on a JVM.
549
If you open a web browser to localhost:8080, you should see the following output:
Hello World!
If you want to generate a native executable directly without using Docker, you can use GraalVM
Native Build Tools. Native Build Tools are plugins shipped by GraalVM for both Maven and Gradle.
You can use them to perform a variety of GraalVM tasks, including generating a native image.
Prerequisites
To build a native image using the Native Build Tools, you’ll need a GraalVM distribution on your
machine. You can either download it manually on the Liberica Native Image Kit page, or you can
use a download manager like SDKMAN!.
To install the native image compiler on macOS or Linux, we recommend using SDKMAN!. Get
SDKMAN! from sdkman.io and install the Liberica GraalVM distribution by using the following
commands:
Verify that the correct version has been configured by checking the output of java -version:
$ java -version
openjdk version "17.0.5" 2022-10-18 LTS
OpenJDK Runtime Environment GraalVM 22.3.0 (build 17.0.5+8-LTS)
OpenJDK 64-Bit Server VM GraalVM 22.3.0 (build 17.0.5+8-LTS, mixed mode)
Windows
On Windows, follow these instructions to install either GraalVM or Liberica Native Image Kit in
version 22.3, the Visual Studio Build Tools and the Windows SDK. Due to the Windows related
command-line maximum length, make sure to use x64 Native Tools Command Prompt instead of
the regular Windows command line to run Maven or Gradle plugins.
Using Maven
As with the buildpack support, you need to make sure that you’re using spring-boot-starter-parent
in order to inherit the native profile and that the org.graalvm.buildtools:native-maven-plugin
plugin is used.
550
With the native profile active, you can invoke the native:compile goal to trigger native-image
compilation:
Using Gradle
When the Native Build Tools Gradle plugin is applied to your project, the Spring Boot Gradle plugin
will automatically trigger the Spring AOT engine. Task dependencies are automatically configured,
so you can just run the standard nativeCompile task to generate a native image:
$ gradle nativeCompile
At this point, your application should work. You can now start the application by running it
directly:
Maven
$ target/myproject
Gradle
$ build/native/nativeCompile/myproject
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v3.1.0-RC2)
....... . . .
....... . . . (log output here)
....... . . .
........ Started MyApplication in 0.08 seconds (process running for 0.095)
The startup time differs from machine to machine, but it should be much faster
NOTE
than a Spring Boot application running on a JVM.
551
If you open a web browser to localhost:8080, you should see the following output:
Hello World!
For native image testing, you’re generally looking to ensure that the following aspects work:
• The Spring AOT engine is able to process your application, and it will run in an AOT-processed
mode.
• GraalVM has enough hints to ensure that a valid native image can be produced.
When a Spring Boot application runs, it attempts to detect if it is running as a native image. If it is
running as a native image, it will initialize the application using the code that was generated during
at build-time by the Spring AOT engine.
If the application is running on a regular JVM, then any AOT generated code is ignored.
Since the native-image compilation phase can take a while to complete, it’s sometimes useful to run
your application on the JVM but have it use the AOT generated initialization code. Doing so helps
you to quickly validate that there are no errors in the AOT generated code and nothing is missing
when your application is eventually converted to a native image.
To run a Spring Boot application on the JVM and have it use AOT generated code you can set the
spring.aot.enabled system property to true.
For example:
You need to ensure that the jar you are testing includes AOT generated code. For
Maven, this means that you should build with -Pnative to activate the native profile.
NOTE
For Gradle, you need to ensure that your build includes the
org.graalvm.buildtools.native plugin.
If your application starts with the spring.aot.enabled property set to true, then you have higher
552
confidence that it will work when converted to a native image.
You can also consider running integration tests against the running application. For example, you
could use the Spring WebClient to call your application REST endpoints. Or you might consider using
a project like Selenium to check your application’s HTML responses.
GraalVM Native Build Tools includes the ability to run tests inside a native image. This can be
helpful when you want to deeply test that the internals of your application work in a GraalVM
native image.
Generating the native image that contains the tests to run can be a time-consuming operation, so
most developers will probably prefer to use the JVM locally. They can, however, be very useful as
part of a CI pipeline. For example, you might choose to run native tests once a day.
Spring Framework includes ahead-of-time support for running tests. All the usual Spring testing
features work with native image tests. For example, you can continue to use the @SpringBootTest
annotation. You can also use Spring Boot test slices to test only specific parts of your application.
• Tests are analyzed in order to discover any ApplicationContext instances that will be required.
• Ahead-of-time processing is applied to each of these application contexts and assets are
generated.
• A native image is created, with the generated assets being processed by GraalVM.
• The native image also includes the JUnit TestEngine configured with a list of the discovered tests.
• The native image is started, triggering the engine which will run each test and report results.
Using Maven
To run native tests using Maven, ensure that your pom.xml file uses the spring-boot-starter-parent.
You should have a <parent> section that looks like this:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>3.1.0-RC2</version>
</parent>
The spring-boot-starter-parent declares a nativeTest profile that configures the executions that are
needed to run the native tests. You can activate profiles using the -P flag on the command line.
553
To build the image and run the tests, use the test goal with the nativeTest profile active:
Using Gradle
The Spring Boot Gradle plugin automatically configures AOT test tasks when the GraalVM Native
Image plugin is applied. You should check that your Gradle build contains a plugins block that
includes org.graalvm.buildtools.native.
To run native tests using Gradle you can use the nativeTest task:
$ gradle nativeTest
Reflection hints are automatically created for configuration properties by the Spring ahead-of-time
engine. Nested configuration properties which are not inner classes, however, must be annotated
with @NestedConfigurationProperty, otherwise they won’t be detected and will not be bindable.
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.NestedConfigurationProperty;
@ConfigurationProperties(prefix = "my.properties")
public class MyProperties {
@NestedConfigurationProperty
private final Nested nested = new Nested();
554
where Nested is:
When using constructor binding, you have to annotate the field with @NestedConfigurationProperty:
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.NestedConfigurationProperty;
@ConfigurationProperties(prefix = "my.properties")
public class MyPropertiesCtor {
@NestedConfigurationProperty
private final Nested nested;
555
When using records, you have to annotate the parameter with @NestedConfigurationProperty:
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.NestedConfigurationProperty;
@ConfigurationProperties(prefix = "my.properties")
public record MyPropertiesRecord(String name, @NestedConfigurationProperty Nested
nested) {
When using Kotlin, you need to annotate the parameter of a data class with
@NestedConfigurationProperty:
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.context.properties.NestedConfigurationProperty
@ConfigurationProperties(prefix = "my.properties")
data class MyPropertiesKotlin(
val name: String,
@NestedConfigurationProperty val nested: Nested
)
Please use public getters and setters in all cases, otherwise the properties will not be
NOTE
bindable.
It is possible to convert a Spring Boot executable jar into a native image as long as the jar contains
the AOT generated assets. This can be useful for a number of reasons, including:
• You can keep your regular JVM pipeline and turn the JVM application into a native image on
your CI/CD platform.
• As native-image does not support cross-compilation, you can keep an OS neutral deployment
artifact which you convert later to different OS architectures.
You can convert a Spring Boot executable jar into a native image using Cloud Native Buildpacks, or
using the native-image tool that is shipped with GraalVM.
Your executable jar must include AOT generated assets such as generated classes
NOTE
and JSON hint files.
Using Buildpacks
Spring Boot applications usually use Cloud Native Buildpacks through the Maven (mvn spring-
boot:build-image) or Gradle (gradle bootBuildImage) integrations. You can, however, also use pack to
turn an AOT processed Spring Boot executable jar into a native container image.
556
First, make sure that a Docker daemon is available (see Get Docker for more details). Configure it to
allow non-root user if you are on Linux.
You also need to install pack by following the installation guide on buildpacks.io.
You do not need to have a local GraalVM installation to generate an image in this
NOTE
way.
Once pack has finished, you can launch the application using docker run:
Another option to turn an AOT processed Spring Boot executable jar into a native executable is to
use the GraalVM native-image tool. For this to work, you’ll need a GraalVM distribution on your
machine. You can either download it manually on the Liberica Native Image Kit page or you can
use a download manager like SDKMAN!.
$ rm -rf target/native
$ mkdir -p target/native
$ cd target/native
$ jar -xvf ../myproject-0.0.1-SNAPSHOT.jar
$ native-image -H:Name=myproject @META-INF/native-image/argfile -cp .:BOOT-
INF/classes:`find BOOT-INF/lib | tr '\n' ':'`
$ mv myproject ../
These commands work on Linux or macOS machines, but you will need to adapt
NOTE
them for Windows.
557
The native-image -cp flag does not accept wildcards. You need to ensure that all
WARNING
jars are listed (the command above uses find and tr to do this).
The GraalVM native image tracing agent allows you to intercept reflection, resources or proxy
usage on the JVM in order to generate the related hints. Spring should generate most of these hints
automatically, but the tracing agent can be used to quickly identify the missing entries.
When using the agent to generate hints for a native image, there are a couple of approaches:
The first option is interesting for identifying the missing hints when a library or a pattern is not
recognized by Spring.
The second option sounds more appealing for a repeatable setup, but by default the generated hints
will include anything required by the test infrastructure. Some of these will be unnecessary when
the application runs for real. To address this problem the agent supports an access-filter file that
will cause certain data to be excluded from the generated output.
Use the following command to launch the application with the native image tracing agent attached:
$ java -Dspring.aot.enabled=true \
-agentlib:native-image-agent=config-output-dir=/path/to/config-dir/ \
-jar target/myproject-0.0.1-SNAPSHOT.jar
Now you can exercise the code paths you want to have hints for and then stop the application with
ctrl-c.
On application shutdown the native image tracing agent will write the hint files to the given config
output directory. You can either manually inspect these files, or use them as input to the native
image build process. To use them as input, copy them into the src/main/resources/META-INF/native-
image/ directory. The next time you build the native image, GraalVM will take these files into
consideration.
There are more advanced options which can be set on the native image tracing agent, for example
filtering the recorded hints by caller classes, etc. For further reading, please see the official
documentation.
If you need to provide your own hints for reflection, resources, serialization, proxy usage etc. you
can use the RuntimeHintsRegistrar API. Create a class that implements the RuntimeHintsRegistrar
interface, and then make appropriate calls to the provided RuntimeHints instance:
558
import java.lang.reflect.Method;
import org.springframework.aot.hint.ExecutableMode;
import org.springframework.aot.hint.RuntimeHints;
import org.springframework.aot.hint.RuntimeHintsRegistrar;
import org.springframework.util.ReflectionUtils;
@Override
public void registerHints(RuntimeHints hints, ClassLoader classLoader) {
// Register method for reflection
Method method = ReflectionUtils.findMethod(MyClass.class, "sayHello",
String.class);
hints.reflection().registerMethod(method, ExecutableMode.INVOKE);
// Register resources
hints.resources().registerPattern("my-resource.txt");
// Register serialization
hints.serialization().registerType(MySerializableClass.class);
// Register proxy
hints.proxies().registerJdkProxy(MyInterface.class);
}
You can then use @ImportRuntimeHints on any @Configuration class (for example your
@SpringBootApplication annotated application class) to activate those hints.
If you have classes which need binding (mostly needed when serializing or deserializing JSON), you
can use @RegisterReflectionForBinding on any bean. Most of the hints are automatically inferred,
for example when accepting or returning data from a @RestController method. But when you work
with WebClient or RestTemplate directly, you might need to use @RegisterReflectionForBinding.
The RuntimeHintsPredicates API can be used to test your hints. The API provides methods that build
a Predicate that can be used to test a RuntimeHints instance.
559
import org.junit.jupiter.api.Test;
import org.springframework.aot.hint.RuntimeHints;
import org.springframework.aot.hint.predicate.RuntimeHintsPredicates;
import org.springframework.boot.docs.nativeimage.advanced.customhints.MyRuntimeHints;
class MyRuntimeHintsTests {
@Test
void shouldRegisterHints() {
RuntimeHints hints = new RuntimeHints();
new MyRuntimeHints().registerHints(hints, getClass().getClassLoader());
assertThat(RuntimeHintsPredicates.resource().forResource("my-
resource.txt")).accepts(hints);
}
GraalVM native images are an evolving technology and not all libraries provide support. The
GraalVM community is helping by providing reachability metadata for projects that don’t yet ship
their own. Spring itself doesn’t contain hints for 3rd party libraries and instead relies on the
reachability metadata project.
If you encounter problems when generating native images for Spring Boot applications, please
check the Spring Boot with GraalVM page of the Spring Boot wiki. You can also contribute issues to
the spring-aot-smoke-tests project on GitHub which is used to confirm that common application
types are working as expected.
If you find a library which doesn’t work with GraalVM, please raise an issue on the reachability
metadata project.
For known limitations with Spring and GraalVM, please see the Spring Boot wiki.
560
Chapter 16. Spring Boot CLI
The Spring Boot CLI is a command line tool that you can use to bootstrap a new project from
start.spring.io or encode a password.
$ spring
usage: spring [--help] [--version]
<command> [<args>]
shell
Start a nested shell
Common options:
You can type spring help to get more details about any of the supported commands, as shown in the
following example:
561
$ spring help init
spring init - Initialize a new project using Spring Initializr (start.spring.io)
Option Description
------ -----------
-a, --artifact-id <String> Project coordinates; infer archive name (for
example 'test')
-b, --boot-version <String> Spring Boot version (for example '1.2.0.RELEASE')
--build <String> Build system to use (for example 'maven' or
'gradle') (default: maven)
-d, --dependencies <String> Comma-separated list of dependency identifiers to
include in the generated project
--description <String> Project description
-f, --force Force overwrite of existing files
--format <String> Format of the generated content (for example
'build' for a build file, 'project' for a
project archive) (default: project)
-g, --group-id <String> Project coordinates (for example 'org.test')
-j, --java-version <String> Language level (for example '1.8')
-l, --language <String> Programming language (for example 'java')
--list List the capabilities of the service. Use it to
discover the dependencies and the types that are
available
-n, --name <String> Project name; infer application name
-p, --packaging <String> Project packaging (for example 'jar')
--package-name <String> Package name
-t, --type <String> Project type. Not normally needed if you use --
build and/or --format. Check the capabilities of
the service (--list) for more details
--target <String> URL of the service to use (default: https://start.
spring.io)
-v, --version <String> Project version (for example '0.0.1-SNAPSHOT')
-x, --extract Extract the project archive. Inferred if a
location is specified without an extension
examples:
562
The version command provides a quick way to check which version of Spring Boot you are using, as
follows:
$ spring version
Spring CLI v3.1.0-RC2
The init command lets you create a new project by using start.spring.io without leaving the shell,
as shown in the following example:
The preceding example creates a my-project directory with a Maven-based project that uses spring-
boot-starter-web and spring-boot-starter-data-jpa. You can list the capabilities of the service by
using the --list flag, as shown in the following example:
Available dependencies:
-----------------------
actuator - Actuator: Production ready features to help you monitor and manage your
application
...
web - Web: Support for full-stack web development, including Tomcat and spring-webmvc
websocket - Websocket: Support for WebSocket development
ws - WS: Support for Spring Web Services
...
The init command supports many options. See the help output for more details. For instance, the
following command creates a Gradle project that uses Java 17 and war packaging:
563
$ spring init --build=gradle --java-version=17 --dependencies=websocket
--packaging=war sample-app.zip
Using service at https://start.spring.io
Content saved to 'sample-app.zip'
Spring Boot includes command-line completion scripts for the BASH and zsh shells. If you do not
use either of these shells (perhaps you are a Windows user), you can use the shell command to
launch an integrated shell, as shown in the following example:
$ spring shell
Spring Boot (v3.1.0-RC2)
Hit TAB to complete. Type \'help' and hit RETURN for help, and \'exit' to quit.
From inside the embedded shell, you can run other commands directly:
$ version
Spring CLI v3.1.0-RC2
The embedded shell supports ANSI color output as well as tab completion. If you need to run a
native command, you can use the ! prefix. To exit the embedded shell, press ctrl-c.
564
Chapter 17. Build Tool Plugins
Spring Boot provides build tool plugins for Maven and Gradle. The plugins offer a variety of
features, including the packaging of executable jars. This section provides more details on both
plugins as well as some help should you need to extend an unsupported build system. If you are
just getting started, you might want to read “Build Systems” from the “Developing with Spring Boot”
section first.
• API
• API
<project xmlns:ivy="antlib:org.apache.ivy.ant"
xmlns:spring-boot="antlib:org.springframework.boot.ant"
name="myapp" default="build">
...
</project>
You need to remember to start Ant using the -lib option, as shown in the following example:
565
The “Using Spring Boot” section includes a more complete example of using Apache
TIP
Ant with spring-boot-antlib.
Once the spring-boot-antlib namespace has been declared, the following additional tasks are
available:
You can use the exejar task to create a Spring Boot executable jar. The following attributes are
supported by the task:
Element Description
resources One or more Resource Collections describing a set of Resources that should be
added to the content of the created jar file.
lib One or more Resource Collections that should be added to the set of jar
libraries that make up the runtime dependency classpath of the application.
Examples
Specify start-class
<spring-boot:exejar destfile="target/my-application.jar"
classes="target/classes" start-class="com.example.MyApplication">
<resources>
<fileset dir="src/main/resources" />
</resources>
<lib>
<fileset dir="lib" />
</lib>
</spring-boot:exejar>
566
Detect start-class
The findmainclass task is used internally by exejar to locate a class declaring a main. If necessary,
you can also use this task directly in your build. The following attributes are supported:
Examples
The Spring Boot Maven and Gradle plugins both make use of spring-boot-loader-tools to actually
generate jars. If you need to, you may use this library directly.
567
17.4.1. Repackaging Archives
When repackaging an archive, you can include references to dependency files by using the
org.springframework.boot.loader.tools.Libraries interface. We do not provide any concrete
implementations of Libraries here as they are usually build-system-specific.
If you do not use Repackager.setMainClass() to specify a main class, the repackager uses ASM to
read class files and tries to find a suitable class with a public static void main(String[] args)
method. An exception is thrown if more than one candidate is found.
568
Java
import java.io.File;
import java.io.IOException;
import java.util.List;
import org.springframework.boot.loader.tools.Library;
import org.springframework.boot.loader.tools.LibraryCallback;
import org.springframework.boot.loader.tools.LibraryScope;
import org.springframework.boot.loader.tools.Repackager;
569
Kotlin
import org.springframework.boot.loader.tools.Library
import org.springframework.boot.loader.tools.LibraryCallback
import org.springframework.boot.loader.tools.LibraryScope
import org.springframework.boot.loader.tools.Repackager
import java.io.File
import java.io.IOException
class MyBuildTool {
@Throws(IOException::class)
fun build() {
val sourceJarFile: File? = ...
val repackager = Repackager(sourceJarFile)
repackager.setBackupSource(false)
repackager.repackage { callback: LibraryCallback -> getLibraries(callback) }
}
@Throws(IOException::class)
private fun getLibraries(callback: LibraryCallback) {
// Build system specific implementation, callback for each dependency
for (nestedJar in getCompileScopeJars()!!) {
callback.library(Library(nestedJar, LibraryScope.COMPILE))
}
// ...
}
570
Chapter 18. “How-to” Guides
This section provides answers to some common ‘how do I do that…’ questions that often arise when
using Spring Boot. Its coverage is not exhaustive, but it does cover quite a lot.
If you have a specific problem that we do not cover here, you might want to check
stackoverflow.com to see if someone has already provided an answer. This is also a great place to
ask new questions (please use the spring-boot tag).
We are also more than happy to extend this section. If you want to add a ‘how-to’, send us a pull
request.
FailureAnalyzer is a great way to intercept an exception on startup and turn it into a human-
readable message, wrapped in a FailureAnalysis. Spring Boot provides such an analyzer for
application-context-related exceptions, JSR-303 validations, and more. You can also create your
own.
org.springframework.boot.diagnostics.FailureAnalyzer=\
com.example.ProjectConstraintViolationFailureAnalyzer
If you need access to the BeanFactory or the Environment, your FailureAnalyzer can
NOTE
implement BeanFactoryAware or EnvironmentAware respectively.
The Spring Boot auto-configuration tries its best to “do the right thing”, but sometimes things fail,
and it can be hard to tell why.
571
been added) by Spring Boot at runtime.
Many more questions can be answered by looking at the source code and the Javadoc. When
reading the code, remember the following rules of thumb:
• Look for classes called *AutoConfiguration and read their sources. Pay special attention to the
@Conditional* annotations to find out what features they enable and when. Add --debug to the
command line or a System property -Ddebug to get a log on the console of all the auto-
configuration decisions that were made in your app. In a running application with actuator
enabled, look at the conditions endpoint (/actuator/conditions or the JMX equivalent) for the
same information.
• Look for classes that are @ConfigurationProperties (such as ServerProperties) and read from
there the available external configuration options. The @ConfigurationProperties annotation has
a name attribute that acts as a prefix to external properties. Thus, ServerProperties has
prefix="server" and its configuration properties are server.port, server.address, and others. In
a running application with actuator enabled, look at the configprops endpoint.
• Look for uses of the bind method on the Binder to pull configuration values explicitly out of the
Environment in a relaxed manner. It is often used with a prefix.
• Look for @ConditionalOnExpression annotations that switch features on and off in response to
SpEL expressions, normally evaluated with placeholders resolved from the Environment.
The SpringApplication sends some special ApplicationEvents to the listeners (some even before the
context is created) and then registers the listeners for events published by the ApplicationContext as
well. See “Application Events and Listeners” in the ‘Spring Boot features’ section for a complete list.
It is also possible to customize the Environment before the application context is refreshed by using
EnvironmentPostProcessor. Each implementation should be registered in META-INF/spring.factories,
as shown in the following example:
org.springframework.boot.env.EnvironmentPostProcessor=com.example.YourEnvironmentPostP
rocessor
572
The implementation can load arbitrary files and add them to the Environment. For instance, the
following example loads a YAML configuration file from the classpath:
Java
import java.io.IOException;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.env.EnvironmentPostProcessor;
import org.springframework.boot.env.YamlPropertySourceLoader;
import org.springframework.core.env.ConfigurableEnvironment;
import org.springframework.core.env.PropertySource;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;
import org.springframework.util.Assert;
@Override
public void postProcessEnvironment(ConfigurableEnvironment environment,
SpringApplication application) {
Resource path = new ClassPathResource("com/example/myapp/config.yml");
PropertySource<?> propertySource = loadYaml(path);
environment.getPropertySources().addLast(propertySource);
}
573
Kotlin
import org.springframework.boot.SpringApplication
import org.springframework.boot.env.EnvironmentPostProcessor
import org.springframework.boot.env.YamlPropertySourceLoader
import org.springframework.core.env.ConfigurableEnvironment
import org.springframework.core.env.PropertySource
import org.springframework.core.io.ClassPathResource
import org.springframework.core.io.Resource
import org.springframework.util.Assert
import java.io.IOException
The Environment has already been prepared with all the usual property sources that
Spring Boot loads by default. It is therefore possible to get the location of the file from
TIP the environment. The preceding example adds the custom-resource property source at
the end of the list so that a key defined in any of the usual other locations takes
precedence. A custom implementation may define another order.
574
18.1.4. Build an ApplicationContext Hierarchy (Adding a Parent or Root
Context)
You can use the ApplicationBuilder class to create parent/child ApplicationContext hierarchies. See
“Fluent Builder API” in the ‘Spring Boot features’ section for more information.
Not all Spring applications have to be web applications (or web services). If you want to execute
some code in a main method but also bootstrap a Spring application to set up the infrastructure to
use, you can use the SpringApplication features of Spring Boot. A SpringApplication changes its
ApplicationContext class, depending on whether it thinks it needs a web application or not. The first
thing you can do to help it is to leave server-related dependencies (such as the servlet API) off the
classpath. If you cannot do that (for example, you run two applications from the same code base)
then you can explicitly call setWebApplicationType(WebApplicationType.NONE) on your
SpringApplication instance or set the applicationContextClass property (through the Java API or
with external properties). Application code that you want to run as your business logic can be
implemented as a CommandLineRunner and dropped into the context as a @Bean definition.
Rather than hardcoding some properties that are also specified in your project’s build
configuration, you can automatically expand them by instead using the existing build
configuration. This is possible in both Maven and Gradle.
You can automatically expand properties from the Maven project by using resource filtering. If you
use the spring-boot-starter-parent, you can then refer to your Maven ‘project properties’ with @..@
placeholders, as shown in the following example:
Properties
app.encoding=@project.build.sourceEncoding@
app.java.version=@java.version@
Yaml
app:
encoding: "@project.build.sourceEncoding@"
java:
version: "@java.version@"
575
Only production configuration is filtered that way (in other words, no filtering is
NOTE
applied on src/test/resources).
If you enable the addResources flag, the spring-boot:run goal can add
src/main/resources directly to the classpath (for hot reloading purposes). Doing so
TIP circumvents the resource filtering and this feature. Instead, you can use the exec:java
goal or customize the plugin’s configuration. See the plugin usage page for more
details.
If you do not use the starter parent, you need to include the following element inside the <build/>
element of your pom.xml:
<resources>
<resource>
<directory>src/main/resources</directory>
<filtering>true</filtering>
</resource>
</resources>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-resources-plugin</artifactId>
<version>2.7</version>
<configuration>
<delimiters>
<delimiter>@</delimiter>
</delimiters>
<useDefaultDelimiters>false</useDefaultDelimiters>
</configuration>
</plugin>
You can automatically expand properties from the Gradle project by configuring the Java plugin’s
processResources task to do so, as shown in the following example:
tasks.named('processResources') {
expand(project.properties)
}
576
You can then refer to your Gradle project’s properties by using placeholders, as shown in the
following example:
Properties
app.name=${name}
app.description=${description}
Yaml
app:
name: "${name}"
description: "${description}"
A SpringApplication has bean property setters, so you can use its Java API as you create the
application to modify its behavior. Alternatively, you can externalize the configuration by setting
properties in spring.main.*. For example, in application.properties, you might have the following
settings:
Properties
spring.main.web-application-type=none
spring.main.banner-mode=off
Yaml
spring:
main:
web-application-type: "none"
banner-mode: "off"
Then the Spring Boot banner is not printed on startup, and the application is not starting an
embedded web server.
Properties defined in external configuration override and replace the values specified with the Java
API, with the notable exception of the primary sources. Primary sources are those provided to the
SpringApplication constructor:
577
Java
import org.springframework.boot.Banner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
Kotlin
import org.springframework.boot.Banner
import org.springframework.boot.SpringApplication
import org.springframework.boot.autoconfigure.SpringBootApplication
@SpringBootApplication
object MyApplication {
@JvmStatic
fun main(args: Array<String>) {
val application = SpringApplication(MyApplication::class.java)
application.setBannerMode(Banner.Mode.OFF)
application.run(*args)
}
578
Java
import org.springframework.boot.Banner;
import org.springframework.boot.builder.SpringApplicationBuilder;
Kotlin
import org.springframework.boot.Banner
import org.springframework.boot.builder.SpringApplicationBuilder
object MyApplication {
@JvmStatic
fun main(args: Array<String>) {
SpringApplicationBuilder()
.bannerMode(Banner.Mode.OFF)
.sources(MyApplication::class.java)
.run(*args)
}
Properties
spring.main.sources=com.example.MyDatabaseConfig,com.example.MyJmsConfig
spring.main.banner-mode=console
Yaml
spring:
main:
sources: "com.example.MyDatabaseConfig,com.example.MyJmsConfig"
banner-mode: "console"
The actual application will show the banner (as overridden by configuration) and uses three
sources for the ApplicationContext. The application sources are:
579
1. MyApplication (from the code)
By default, properties from different sources are added to the Spring Environment in a defined order
(see “Externalized Configuration” in the ‘Spring Boot features’ section for the exact order).
You can also provide the following System properties (or environment variables) to change the
behavior:
No matter what you set in the environment, Spring Boot always loads application.properties as
described above. By default, if YAML is used, then files with the ‘.yaml’ and ‘.yml’ extension are also
added to the list.
If you want detailed information about the files that are being loaded you can set the
TIP
logging level of org.springframework.boot.context.config to trace.
Some people like to use (for example) --port=9000 instead of --server.port=9000 to set configuration
properties on the command line. You can enable this behavior by using placeholders in
application.properties, as shown in the following example:
Properties
server.port=${port:8080}
Yaml
server:
port: "${port:8080}"
If you inherit from the spring-boot-starter-parent POM, the default filter token of the
maven-resources-plugins has been changed from ${*} to @ (that is, @maven.token@
TIP instead of ${maven.token}) to prevent conflicts with Spring-style placeholders. If you
have enabled Maven filtering for the application.properties directly, you may want to
also change the default filter token to use other delimiters.
580
In this specific case, the port binding works in a PaaS environment such as Heroku
or Cloud Foundry. In those two platforms, the PORT environment variable is set
NOTE
automatically and Spring can bind to capitalized synonyms for Environment
properties.
YAML is a superset of JSON and, as such, is a convenient syntax for storing external properties in a
hierarchical format, as shown in the following example:
spring:
application:
name: "cruncher"
datasource:
driver-class-name: "com.mysql.jdbc.Driver"
url: "jdbc:mysql://localhost/test"
server:
port: 9000
Create a file called application.yaml and put it in the root of your classpath. Then add snakeyaml to
your dependencies (Maven coordinates org.yaml:snakeyaml, already included if you use the spring-
boot-starter). A YAML file is parsed to a Java Map<String,Object> (like a JSON object), and Spring
Boot flattens the map so that it is one level deep and has period-separated keys, as many people are
used to with Properties files in Java.
spring.application.name=cruncher
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost/test
server.port=9000
See “Working With YAML” in the ‘Spring Boot features’ section for more information about YAML.
The Spring Environment has an API for this, but you would normally set a System property
(spring.profiles.active) or an OS environment variable (SPRING_PROFILES_ACTIVE). Also, you can
launch your application with a -D argument (remember to put it before the main class or jar
archive), as follows:
In Spring Boot, you can also set the active profile in application.properties, as shown in the
following example:
581
Properties
spring.profiles.active=production
Yaml
spring:
profiles:
active: "production"
A value set this way is replaced by the System property or environment variable setting but not by
the SpringApplicationBuilder.profiles() method. Thus, the latter Java API can be used to augment
the profiles without changing the defaults.
See “Profiles” in the “Spring Boot features” section for more information.
The default profile is a profile that is enabled if no profile is active. By default, the name of the
default profile is default, but it could be changed using a System property (spring.profiles.default)
or an OS environment variable (SPRING_PROFILES_DEFAULT).
In Spring Boot, you can also set the default profile name in application.properties, as shown in the
following example:
Properties
spring.profiles.default=dev
Yaml
spring:
profiles:
default: "dev"
See “Profiles” in the “Spring Boot features” section for more information.
Spring Boot supports multi-document YAML and Properties files (see Working With Multi-
Document Files for details) which can be activated conditionally based on the active profiles.
582
Properties
server.port=9000
#---
spring.config.activate.on-profile=development
server.port=9001
#---
spring.config.activate.on-profile=production
server.port=0
Yaml
server:
port: 9000
---
spring:
config:
activate:
on-profile: "development"
server:
port: 9001
---
spring:
config:
activate:
on-profile: "production"
server:
port: 0
In the preceding example, the default port is 9000. However, if the Spring profile called
‘development’ is active, then the port is 9001. If ‘production’ is active, then the port is 0.
The documents are merged in the order in which they are encountered. Later
NOTE
values override earlier values.
Spring Boot binds external properties from application.properties (or YAML files and other places)
into an application at runtime. There is not (and technically cannot be) an exhaustive list of all
supported properties in a single location, because contributions can come from additional jar files
on your classpath.
A running application with the Actuator features has a configprops endpoint that shows all the
bound and bindable properties available through @ConfigurationProperties.
The appendix includes an application.properties example with a list of the most common
properties supported by Spring Boot. The definitive list comes from searching the source code for
@ConfigurationProperties and @Value annotations as well as the occasional use of Binder. For more
about the exact ordering of loading properties, see "Externalized Configuration".
583
18.3. Embedded Web Servers
Each Spring Boot web application includes an embedded web server. This feature leads to a
number of how-to questions, including how to change the embedded server and how to configure
the embedded server. This section answers those questions.
• For servlet stack applications, the spring-boot-starter-web includes Tomcat by including spring-
boot-starter-tomcat, but you can use spring-boot-starter-jetty or spring-boot-starter-
undertow instead.
When switching to a different HTTP server, you need to swap the default dependencies for those
that you need instead. To help with this process, Spring Boot provides a separate starter for each of
the supported HTTP servers.
The following Maven example shows how to exclude Tomcat and include Jetty for Spring MVC:
<properties>
<servlet-api.version>3.1.0</servlet-api.version>
</properties>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<exclusions>
<!-- Exclude the Tomcat dependency -->
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
</exclusion>
</exclusions>
</dependency>
<!-- Use Jetty instead -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jetty</artifactId>
</dependency>
The version of the servlet API has been overridden as, unlike Tomcat 9 and
NOTE
Undertow 2, Jetty 9.4 does not support servlet 4.0.
If you wish to use Jetty 10, which does support servlet 4.0, you can do so as shown in the following
example:
584
<properties>
<jetty.version>10.0.8</jetty.version>
</properties>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<exclusions>
<!-- Exclude the Tomcat dependency -->
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
</exclusion>
</exclusions>
</dependency>
<!-- Use Jetty instead -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jetty</artifactId>
<exclusions>
<!-- Exclude the Jetty-9 specific dependencies -->
<exclusion>
<groupId>org.eclipse.jetty.websocket</groupId>
<artifactId>websocket-server</artifactId>
</exclusion>
<exclusion>
<groupId>org.eclipse.jetty.websocket</groupId>
<artifactId>javax-websocket-server-impl</artifactId>
</exclusion>
</exclusions>
</dependency>
Note that along with excluding the Tomcat starter, a couple of Jetty9-specific dependencies also
need to be excluded.
The following Gradle example configures the necessary dependencies and a module replacement to
use Undertow in place of Reactor Netty for Spring WebFlux:
dependencies {
implementation "org.springframework.boot:spring-boot-starter-undertow"
implementation "org.springframework.boot:spring-boot-starter-webflux"
modules {
module("org.springframework.boot:spring-boot-starter-reactor-netty") {
replacedBy("org.springframework.boot:spring-boot-starter-undertow", "Use
Undertow instead of Reactor Netty")
}
}
}
585
spring-boot-starter-reactor-netty is required to use the WebClient class, so you may
NOTE need to keep a dependency on Netty even when you need to include a different
HTTP server.
If your classpath contains the necessary bits to start a web server, Spring Boot will automatically
start it. To disable this behavior configure the WebApplicationType in your application.properties, as
shown in the following example:
Properties
spring.main.web-application-type=none
Yaml
spring:
main:
web-application-type: "none"
In a standalone application, the main HTTP port defaults to 8080 but can be set with server.port (for
example, in application.properties or as a System property). Thanks to relaxed binding of
Environment values, you can also use SERVER_PORT (for example, as an OS environment variable).
To switch off the HTTP endpoints completely but still create a WebApplicationContext, use
server.port=-1 (doing so is sometimes useful for testing).
For more details, see “Customizing Embedded Servlet Containers” in the ‘Spring Boot Features’
section, or the ServerProperties source code.
To scan for a free port (using OS natives to prevent clashes) use server.port=0.
You can access the port the server is running on from log output or from the
WebServerApplicationContext through its WebServer. The best way to get that and be sure it has been
initialized is to add a @Bean of type ApplicationListener<WebServerInitializedEvent> and pull the
container out of the event when it is published.
586
Java
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.boot.test.web.server.LocalServerPort;
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyWebIntegrationTests {
@LocalServerPort
int port;
// ...
Kotlin
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment
import org.springframework.boot.test.web.server.LocalServerPort
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyWebIntegrationTests {
@LocalServerPort
var port = 0
// ...
HTTP response compression is supported by Jetty, Tomcat, Reactor Netty, and Undertow. It can be
enabled in application.properties, as follows:
Properties
server.compression.enabled=true
587
Yaml
server:
compression:
enabled: true
By default, responses must be at least 2048 bytes in length for compression to be performed. You
can configure this behavior by setting the server.compression.min-response-size property.
By default, responses are compressed only if their content type is one of the following:
• text/html
• text/xml
• text/plain
• text/css
• text/javascript
• application/javascript
• application/json
• application/xml
SSL can be configured declaratively by setting the various server.ssl.* properties, typically in
application.properties or application.yaml. The following example shows setting SSL properties
using a Java KeyStore file:
Properties
server.port=8443
server.ssl.key-store=classpath:keystore.jks
server.ssl.key-store-password=secret
server.ssl.key-password=another-secret
Yaml
server:
port: 8443
ssl:
key-store: "classpath:keystore.jks"
key-store-password: "secret"
key-password: "another-secret"
The following example shows setting SSL properties using PEM-encoded certificate and private key
files:
588
Properties
server.port=8443
server.ssl.certificate=classpath:my-cert.crt
server.ssl.certificate-private-key=classpath:my-cert.key
server.ssl.trust-certificate=classpath:ca-cert.crt
Yaml
server:
port: 8443
ssl:
certificate: "classpath:my-cert.crt"
certificate-private-key: "classpath:my-cert.key"
trust-certificate: "classpath:ca-cert.crt"
Alternatively, the SSL trust material can be configured in an SSL bundle and applied to the web
server as shown in this example:
Properties
server.port=8443
server.ssl.bundle=example
Yaml
server:
port: 8443
ssl:
bundle: "example"
Using configuration such as the preceding example means the application no longer supports a
plain HTTP connector at port 8080. Spring Boot does not support the configuration of both an HTTP
connector and an HTTPS connector through application.properties. If you want to have both, you
need to configure one of them programmatically. We recommend using application.properties to
configure HTTPS, as the HTTP connector is the easier of the two to configure programmatically.
You can enable HTTP/2 support in your Spring Boot application with the server.http2.enabled
configuration property. Both h2 (HTTP/2 over TLS) and h2c (HTTP/2 over TCP) are supported. To use
h2, SSL must also be enabled. When SSL is not enabled, h2c will be used. You may, for example, want
to use h2c when your application is running behind a proxy server that is performing TLS
termination.
589
HTTP/2 With Tomcat
Spring Boot ships by default with Tomcat 10.1.x which supports h2c and h2 out of the box.
Alternatively, you can use libtcnative for h2 support if the library and its dependencies are
installed on the host operating system.
The library directory must be made available, if not already, to the JVM library path. You can do so
with a JVM argument such as -Djava.library.path=/usr/local/opt/tomcat-native/lib. More on this
in the official Tomcat documentation.
Spring Boot manages the version for the io.netty:netty-tcnative-boringssl-static "uber jar",
containing native libraries for all platforms. Developers can choose to import only the required
dependencies using a classifier (see the Netty official documentation).
Generally, you should first consider using one of the many available configuration keys and
customize your web server by adding new entries in your application.properties or
application.yaml file. See “Discover Built-in Options for External Properties”). The server.*
namespace is quite useful here, and it includes namespaces like server.tomcat.*, server.jetty.*
and others, for server-specific features. See the list of Common Application Properties.
The previous sections covered already many common use cases, such as compression, SSL or
HTTP/2. However, if a configuration key does not exist for your use case, you should then look at
WebServerFactoryCustomizer. You can declare such a component and get access to the server factory
relevant to your choice: you should select the variant for the chosen Server (Tomcat, Jetty, Reactor
Netty, Undertow) and the chosen web stack (servlet or reactive).
The example below is for Tomcat with the spring-boot-starter-web (servlet stack):
590
Java
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.stereotype.Component;
@Component
public class MyTomcatWebServerCustomizer implements
WebServerFactoryCustomizer<TomcatServletWebServerFactory> {
@Override
public void customize(TomcatServletWebServerFactory factory) {
// customize the factory here
}
Kotlin
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory
import org.springframework.boot.web.server.WebServerFactoryCustomizer
import org.springframework.stereotype.Component
@Component
class MyTomcatWebServerCustomizer :
WebServerFactoryCustomizer<TomcatServletWebServerFactory?> {
Spring Boot uses that infrastructure internally to auto-configure the server. Auto-
configured WebServerFactoryCustomizer beans have an order of 0 and will be
NOTE
processed before any user-defined customizers, unless it has an explicit order that
states otherwise.
Once you have got access to a WebServerFactory using the customizer, you can use it to configure
specific parts, like connectors, server resources, or the server itself - all using server-specific APIs.
591
Server Servlet stack Reactive stack
As a last resort, you can also declare your own WebServerFactory bean, which will override the one
provided by Spring Boot. When you do so, auto-configured customizers are still applied on your
custom factory, so use that option carefully.
In a servlet stack application, that is with the spring-boot-starter-web, there are two ways to add
Servlet, Filter, ServletContextListener, and the other listeners supported by the Servlet API to your
application:
To add a Servlet, Filter, or servlet *Listener by using a Spring bean, you must provide a @Bean
definition for it. Doing so can be very useful when you want to inject configuration or
dependencies. However, you must be very careful that they do not cause eager initialization of too
many other beans, because they have to be installed in the container very early in the application
lifecycle. (For example, it is not a good idea to have them depend on your DataSource or JPA
configuration.) You can work around such restrictions by initializing the beans lazily when first
used instead of on initialization.
In the case of filters and servlets, you can also add mappings and init parameters by adding a
FilterRegistrationBean or a ServletRegistrationBean instead of or in addition to the underlying
component.
Like any other Spring bean, you can define the order of servlet filter beans; please make sure to
check the “Registering Servlets, Filters, and Listeners as Spring Beans” section.
As described earlier, any Servlet or Filter beans are registered with the servlet container
automatically. To disable registration of a particular Filter or Servlet bean, create a registration
bean for it and mark it as disabled, as shown in the following example:
592
Java
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyFilterConfiguration {
@Bean
public FilterRegistrationBean<MyFilter> registration(MyFilter filter) {
FilterRegistrationBean<MyFilter> registration = new
FilterRegistrationBean<>(filter);
registration.setEnabled(false);
return registration;
}
Kotlin
import org.springframework.boot.web.servlet.FilterRegistrationBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyFilterConfiguration {
@Bean
fun registration(filter: MyFilter): FilterRegistrationBean<MyFilter> {
val registration = FilterRegistrationBean(filter)
registration.isEnabled = false
return registration
}
@WebServlet, @WebFilter, and @WebListener annotated classes can be automatically registered with an
embedded servlet container by annotating a @Configuration class with @ServletComponentScan and
specifying the package(s) containing the components that you want to register. By default,
@ServletComponentScan scans from the package of the annotated class.
Access logs can be configured for Tomcat, Undertow, and Jetty through their respective
namespaces.
593
For instance, the following settings log access on Tomcat with a custom pattern.
Properties
server.tomcat.basedir=my-tomcat
server.tomcat.accesslog.enabled=true
server.tomcat.accesslog.pattern=%t %a %r %s (%D ms)
Yaml
server:
tomcat:
basedir: "my-tomcat"
accesslog:
enabled: true
pattern: "%t %a %r %s (%D ms)"
The default location for logs is a logs directory relative to the Tomcat base directory.
By default, the logs directory is a temporary directory, so you may want to fix
NOTE Tomcat’s base directory or use an absolute path for the logs. In the preceding
example, the logs are available in my-tomcat/logs relative to the working directory
of the application.
Access logging for Undertow can be configured in a similar fashion, as shown in the following
example:
Properties
server.undertow.accesslog.enabled=true
server.undertow.accesslog.pattern=%t %a %r %s (%D ms)
server.undertow.options.server.record-request-start-time=true
Yaml
server:
undertow:
accesslog:
enabled: true
pattern: "%t %a %r %s (%D ms)"
options:
server:
record-request-start-time: true
Note that, in addition to enabling access logging and configuring its pattern, recording request start
times has also been enabled. This is required when including the response time (%D) in the access
log pattern. Logs are stored in a logs directory relative to the working directory of the application.
You can customize this location by setting the server.undertow.accesslog.dir property.
594
Finally, access logging for Jetty can also be configured as follows:
Properties
server.jetty.accesslog.enabled=true
server.jetty.accesslog.filename=/var/log/jetty-access.log
Yaml
server:
jetty:
accesslog:
enabled: true
filename: "/var/log/jetty-access.log"
By default, logs are redirected to System.err. For more details, see the Jetty documentation.
If your application is running behind a proxy, a load-balancer or in the cloud, the request
information (like the host, port, scheme…) might change along the way. Your application may be
running on 10.10.10.10:8080, but HTTP clients should only see example.org.
RFC7239 "Forwarded Headers" defines the Forwarded HTTP header; proxies can use this header to
provide information about the original request. You can configure your application to read those
headers and automatically use that information when creating links and sending them to clients in
HTTP 302 responses, JSON documents or HTML pages. There are also non-standard headers, like X-
Forwarded-Host, X-Forwarded-Port, X-Forwarded-Proto, X-Forwarded-Ssl, and X-Forwarded-Prefix.
If the proxy adds the commonly used X-Forwarded-For and X-Forwarded-Proto headers, setting
server.forward-headers-strategy to NATIVE is enough to support those. With this option, the Web
servers themselves natively support this feature; you can check their specific documentation to
learn about specific behavior.
If this is not enough, Spring Framework provides a ForwardedHeaderFilter. You can register it as a
servlet filter in your application by setting server.forward-headers-strategy is set to FRAMEWORK.
If you are using Tomcat and terminating SSL at the proxy, server.tomcat.redirect-
TIP context-root should be set to false. This allows the X-Forwarded-Proto header to be
honored before any redirects are performed.
If you use Tomcat, you can additionally configure the names of the headers used to carry
“forwarded” information, as shown in the following example:
595
Properties
server.tomcat.remoteip.remote-ip-header=x-your-remote-ip-header
server.tomcat.remoteip.protocol-header=x-your-protocol-header
Yaml
server:
tomcat:
remoteip:
remote-ip-header: "x-your-remote-ip-header"
protocol-header: "x-your-protocol-header"
Tomcat is also configured with a regular expression that matches internal proxies that are to be
trusted. See the server.tomcat.remoteip.internal-proxies entry in the appendix for its default value.
You can customize the valve’s configuration by adding an entry to application.properties, as shown
in the following example:
Properties
server.tomcat.remoteip.internal-proxies=192\\.168\\.\\d{1,3}\\.\\d{1,3}
Yaml
server:
tomcat:
remoteip:
internal-proxies: "192\\.168\\.\\d{1,3}\\.\\d{1,3}"
You can trust all proxies by setting the internal-proxies to empty (but do not do so
NOTE
in production).
You can take complete control of the configuration of Tomcat’s RemoteIpValve by switching the
automatic one off (to do so, set server.forward-headers-strategy=NONE) and adding a new valve
instance using a WebServerFactoryCustomizer bean.
596
Java
import org.apache.catalina.connector.Connector;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyTomcatConfiguration {
@Bean
public WebServerFactoryCustomizer<TomcatServletWebServerFactory>
connectorCustomizer() {
return (tomcat) -> tomcat.addAdditionalTomcatConnectors(createConnector());
}
597
Kotlin
import org.apache.catalina.connector.Connector
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory
import org.springframework.boot.web.server.WebServerFactoryCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyTomcatConfiguration {
@Bean
fun connectorCustomizer():
WebServerFactoryCustomizer<TomcatServletWebServerFactory> {
return WebServerFactoryCustomizer { tomcat: TomcatServletWebServerFactory ->
tomcat.addAdditionalTomcatConnectors(
createConnector()
)
}
}
Embedded Tomcat’s MBean registry is disabled by default. This minimizes Tomcat’s memory
footprint. If you want to use Tomcat’s MBeans, for example so that they can be used by Micrometer
to expose metrics, you must use the server.tomcat.mbeanregistry.enabled property to do so, as
shown in the following example:
Properties
server.tomcat.mbeanregistry.enabled=true
Yaml
server:
tomcat:
mbeanregistry:
enabled: true
598
18.3.15. Enable Multiple Listeners with Undertow
Java
import io.undertow.Undertow.Builder;
import org.springframework.boot.web.embedded.undertow.UndertowServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyUndertowConfiguration {
@Bean
public WebServerFactoryCustomizer<UndertowServletWebServerFactory>
undertowListenerCustomizer() {
return (factory) -> factory.addBuilderCustomizers(this::addHttpListener);
}
599
Kotlin
import io.undertow.Undertow
import org.springframework.boot.web.embedded.undertow.UndertowBuilderCustomizer
import org.springframework.boot.web.embedded.undertow.UndertowServletWebServerFactory
import org.springframework.boot.web.server.WebServerFactoryCustomizer
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyUndertowConfiguration {
@Bean
fun undertowListenerCustomizer():
WebServerFactoryCustomizer<UndertowServletWebServerFactory> {
return WebServerFactoryCustomizer { factory: UndertowServletWebServerFactory
->
factory.addBuilderCustomizers(
UndertowBuilderCustomizer { builder: Undertow.Builder ->
addHttpListener(builder) })
}
}
If you want to use @ServerEndpoint in a Spring Boot application that used an embedded container,
you must declare a single ServerEndpointExporter @Bean, as shown in the following example:
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;
@Configuration(proxyBeanMethods = false)
public class MyWebSocketConfiguration {
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
600
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.web.socket.server.standard.ServerEndpointExporter
@Configuration(proxyBeanMethods = false)
class MyWebSocketConfiguration {
@Bean
fun serverEndpointExporter(): ServerEndpointExporter {
return ServerEndpointExporter()
}
The bean shown in the preceding example registers any @ServerEndpoint annotated beans with the
underlying WebSocket container. When deployed to a standalone servlet container, this role is
performed by a servlet container initializer, and the ServerEndpointExporter bean is not required.
Any Spring @RestController in a Spring Boot application should render JSON response by default as
long as Jackson2 is on the classpath, as shown in the following example:
Java
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class MyController {
@RequestMapping("/thing")
public MyThing thing() {
return new MyThing();
}
601
Kotlin
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController
@RestController
class MyController {
@RequestMapping("/thing")
fun thing(): MyThing {
return MyThing()
}
As long as MyThing can be serialized by Jackson2 (true for a normal POJO or Groovy object), then
localhost:8080/thing serves a JSON representation of it by default. Note that, in a browser, you
might sometimes see XML responses, because browsers tend to send accept headers that prefer
XML.
If you have the Jackson XML extension (jackson-dataformat-xml) on the classpath, you can use it to
render XML responses. The previous example that we used for JSON would work. To use the
Jackson XML renderer, add the following dependency to your project:
<dependency>
<groupId>com.fasterxml.jackson.dataformat</groupId>
<artifactId>jackson-dataformat-xml</artifactId>
</dependency>
If Jackson’s XML extension is not available and JAXB is available, XML can be rendered with the
additional requirement of having MyThing annotated as @XmlRootElement, as shown in the following
example:
602
Java
import jakarta.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class MyThing {
Kotlin
import jakarta.xml.bind.annotation.XmlRootElement
@XmlRootElement
class MyThing {
You will need to ensure that the JAXB library is part of your project, for example by adding:
<dependency>
<groupId>org.glassfish.jaxb</groupId>
<artifactId>jaxb-runtime</artifactId>
</dependency>
To get the server to render XML instead of JSON, you might have to send an Accept:
NOTE
text/xml header (or use a browser).
Spring MVC (client and server side) uses HttpMessageConverters to negotiate content conversion in
an HTTP exchange. If Jackson is on the classpath, you already get the default converter(s) provided
by Jackson2ObjectMapperBuilder, an instance of which is auto-configured for you.
The ObjectMapper (or XmlMapper for Jackson XML converter) instance (created by default) has the
following customized properties:
603
• MapperFeature.DEFAULT_VIEW_INCLUSION is disabled
• DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES is disabled
• SerializationFeature.WRITE_DATES_AS_TIMESTAMPS is disabled
Spring Boot also has some features to make it easier to customize this behavior.
You can configure the ObjectMapper and XmlMapper instances by using the environment. Jackson
provides an extensive suite of on/off features that can be used to configure various aspects of its
processing. These features are described in six enums (in Jackson) that map onto properties in the
environment:
If you want to replace the default ObjectMapper completely, either define a @Bean of that type and
mark it as @Primary or, if you prefer the builder-based approach, define a
Jackson2ObjectMapperBuilder @Bean. Note that, in either case, doing so disables all auto-configuration
of the ObjectMapper.
604
If you provide any @Beans of type MappingJackson2HttpMessageConverter, they replace the default
value in the MVC configuration. Also, a convenience bean of type HttpMessageConverters is provided
(and is always available if you use the default MVC configuration). It has some useful methods to
access the default and user-enhanced message converters.
See the “Customize the @ResponseBody Rendering” section and the WebMvcAutoConfiguration source
code for more details.
Spring uses HttpMessageConverters to render @ResponseBody (or responses from @RestController). You
can contribute additional converters by adding beans of the appropriate type in a Spring Boot
context. If a bean you add is of a type that would have been included by default anyway (such as
MappingJackson2HttpMessageConverter for JSON conversions), it replaces the default value. A
convenience bean of type HttpMessageConverters is provided and is always available if you use the
default MVC configuration. It has some useful methods to access the default and user-enhanced
message converters (For example, it can be useful if you want to manually inject them into a
custom RestTemplate).
As in normal MVC usage, any WebMvcConfigurer beans that you provide can also contribute
converters by overriding the configureMessageConverters method. However, unlike with normal
MVC, you can supply only additional converters that you need (because Spring Boot uses the same
mechanism to contribute its defaults). Finally, if you opt out of the Spring Boot default MVC
configuration by providing your own @EnableWebMvc configuration, you can take control completely
and do everything manually by using getMessageConverters from WebMvcConfigurationSupport.
Spring Boot embraces the servlet 5 jakarta.servlet.http.Part API to support uploading files. By
default, Spring Boot configures Spring MVC with a maximum size of 1MB per file and a maximum
of 10MB of file data in a single request. You may override these values, the location to which
intermediate data is stored (for example, to the /tmp directory), and the threshold past which data is
flushed to disk by using the properties exposed in the MultipartProperties class. For example, if you
want to specify that files be unlimited, set the spring.servlet.multipart.max-file-size property to
-1.
The multipart support is helpful when you want to receive multipart encoded file data as a
@RequestParam-annotated parameter of type MultipartFile in a Spring MVC controller handler
method.
605
18.4.6. Switch Off the Spring MVC DispatcherServlet
By default, all content is served from the root of your application (/). If you would rather map to a
different path, you can configure one as follows:
Properties
spring.mvc.servlet.path=/mypath
Yaml
spring:
mvc:
servlet:
path: "/mypath"
If you have additional servlets you can declare a @Bean of type Servlet or ServletRegistrationBean
for each and Spring Boot will register them transparently to the container. Because servlets are
registered that way, they can be mapped to a sub-context of the DispatcherServlet without invoking
it.
Configuring the DispatcherServlet yourself is unusual but if you really need to do it, a @Bean of type
DispatcherServletPath must be provided as well to provide the path of your custom
DispatcherServlet.
The easiest way to take complete control over MVC configuration is to provide your own
@Configuration with the @EnableWebMvc annotation. Doing so leaves all MVC configuration in your
hands.
A ViewResolver is a core component of Spring MVC, translating view names in @Controller to actual
View implementations. Note that ViewResolvers are mainly used in UI applications, rather than
REST-style services (a View is not used to render a @ResponseBody). There are many implementations
of ViewResolver to choose from, and Spring on its own is not opinionated about which ones you
should use. Spring Boot, on the other hand, installs one or two for you, depending on what it finds
on the classpath and in the application context. The DispatcherServlet uses all the resolvers it finds
in the application context, trying each one in turn until it gets a result. If you add your own, you
have to be aware of the order and in which position your resolver is added.
606
spring.mvc.view.suffix). You can override it by providing a bean of the same type.
• If you use Groovy templates (actually, if groovy-templates is on your classpath), you also have a
GroovyMarkupViewResolver named ‘groovyMarkupViewResolver’. It looks for resources in a
loader path by surrounding the view name with a prefix and suffix (externalized to
spring.groovy.template.prefix and spring.groovy.template.suffix). The prefix and suffix have
default values of ‘classpath:/templates/’ and ‘.tpl’, respectively. You can override
GroovyMarkupViewResolver by providing a bean of the same name.
• WebMvcAutoConfiguration
• ThymeleafAutoConfiguration
• FreeMarkerAutoConfiguration
• GroovyTemplateAutoConfiguration
18.5. Jersey
607
18.5.1. Secure Jersey endpoints with Spring Security
Spring Security can be used to secure a Jersey-based web application in much the same way as it
can be used to secure a Spring MVC-based web application. However, if you want to use Spring
Security’s method-level security with Jersey, you must configure Jersey to use setStatus(int) rather
sendError(int). This prevents Jersey from committing the response before Spring Security has had
an opportunity to report an authentication or authorization failure to the client.
import java.util.Collections;
import org.glassfish.jersey.server.ResourceConfig;
import org.springframework.stereotype.Component;
@Component
public class JerseySetStatusOverSendErrorConfig extends ResourceConfig {
public JerseySetStatusOverSendErrorConfig() {
register(Endpoint.class);
setProperties(Collections.singletonMap("jersey.config.server.response.setStatusOverSen
dError", true));
}
To use Jersey alongside another web framework, such as Spring MVC, it should be configured so
that it will allow the other framework to handle requests that it cannot handle. First, configure
Jersey to use a filter rather than a servlet by configuring the spring.jersey.type application
property with a value of filter. Second, configure your ResourceConfig to forward requests that
would have resulted in a 404, as shown in the following example.
608
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.servlet.ServletProperties;
import org.springframework.stereotype.Component;
@Component
public class JerseyConfig extends ResourceConfig {
public JerseyConfig() {
register(Endpoint.class);
property(ServletProperties.FILTER_FORWARD_ON_404, true);
}
The exact details of the proxy configuration depend on the underlying client request factory that is
being used.
609
Java
import io.netty.channel.ChannelOption;
import io.netty.handler.timeout.ReadTimeoutHandler;
import reactor.netty.http.client.HttpClient;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.client.reactive.ClientHttpConnector;
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
import org.springframework.http.client.reactive.ReactorResourceFactory;
@Configuration(proxyBeanMethods = false)
public class MyReactorNettyClientConfiguration {
@Bean
ClientHttpConnector clientHttpConnector(ReactorResourceFactory resourceFactory) {
HttpClient httpClient =
HttpClient.create(resourceFactory.getConnectionProvider())
.runOn(resourceFactory.getLoopResources())
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 60000)
.doOnConnected((connection) -> connection.addHandlerLast(new
ReadTimeoutHandler(60)));
return new ReactorClientHttpConnector(httpClient);
}
610
Kotlin
import io.netty.channel.ChannelOption
import io.netty.handler.timeout.ReadTimeoutHandler
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.http.client.reactive.ClientHttpConnector
import org.springframework.http.client.reactive.ReactorClientHttpConnector
import org.springframework.http.client.reactive.ReactorResourceFactory
import reactor.netty.http.client.HttpClient
@Configuration(proxyBeanMethods = false)
class MyReactorNettyClientConfiguration {
@Bean
fun clientHttpConnector(resourceFactory: ReactorResourceFactory):
ClientHttpConnector {
val httpClient = HttpClient.create(resourceFactory.connectionProvider)
.runOn(resourceFactory.loopResources)
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 60000)
.doOnConnected { connection ->
connection.addHandlerLast(ReadTimeoutHandler(60))
}
return ReactorClientHttpConnector(httpClient)
}
Note the use of ReactorResourceFactory for the connection provider and event loop
TIP resources. This ensures efficient sharing of resources for the server receiving requests
and the client making requests.
18.7. Logging
Spring Boot has no mandatory logging dependency, except for the Commons Logging API, which is
typically provided by Spring Framework’s spring-jcl module. To use Logback, you need to include
it and spring-jcl on the classpath. The recommended way to do that is through the starters, which
all depend on spring-boot-starter-logging. For a web application, you need only spring-boot-
starter-web, since it depends transitively on the logging starter. If you use Maven, the following
dependency adds logging for you:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
Spring Boot has a LoggingSystem abstraction that attempts to configure logging based on the content
611
of the classpath. If Logback is available, it is the first choice.
If the only change you need to make to logging is to set the levels of various loggers, you can do so
in application.properties by using the "logging.level" prefix, as shown in the following example:
Properties
logging.level.org.springframework.web=debug
logging.level.org.hibernate=error
Yaml
logging:
level:
org.springframework.web: "debug"
org.hibernate: "error"
You can also set the location of a file to which to write the log (in addition to the console) by using
logging.file.name.
To configure the more fine-grained settings of a logging system, you need to use the native
configuration format supported by the LoggingSystem in question. By default, Spring Boot picks up
the native configuration from its default location for the system (such as classpath:logback.xml for
Logback), but you can set the location of the config file by using the logging.config property.
If you need to apply customizations to logback beyond those that can be achieved with
application.properties, you will need to add a standard logback configuration file. You can add a
logback.xml file to the root of your classpath for logback to find. You can also use logback-spring.xml
if you want to use the Spring Boot Logback extensions.
The Logback documentation has a dedicated section that covers configuration in some
TIP
detail.
Spring Boot provides a number of logback configurations that can be included in your own
configuration. These includes are designed to allow certain common Spring Boot conventions to be
re-applied.
• defaults.xml - Provides conversion rules, pattern properties and common logger configurations.
In addition, a legacy base.xml file is provided for compatibility with earlier versions of Spring Boot.
612
A typical custom logback.xml file would look something like this:
Your logback configuration file can also make use of System properties that the LoggingSystem takes
care of creating for you:
• ${LOG_PATH}: Whether logging.file.path (representing a directory for log files to live in) was set
in Boot’s external configuration.
Spring Boot also provides some nice ANSI color terminal output on a console (but not in a log file)
by using a custom Logback converter. See the CONSOLE_LOG_PATTERN in the defaults.xml configuration
for an example.
If Groovy is on the classpath, you should be able to configure Logback with logback.groovy as well.
If present, this setting is given preference.
Spring extensions are not supported with Groovy configuration. Any logback-
NOTE
spring.groovy files will not be detected.
If you want to disable console logging and write output only to a file, you need a custom logback-
spring.xml that imports file-appender.xml but not console-appender.xml, as shown in the following
example:
613
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<include resource="org/springframework/boot/logging/logback/defaults.xml" />
<property name="LOG_FILE" value="${LOG_FILE:-${LOG_PATH:-${LOG_TEMP:-
${java.io.tmpdir:-/tmp}}/}spring.log}"/>
<include resource="org/springframework/boot/logging/logback/file-appender.xml" />
<root level="INFO">
<appender-ref ref="FILE" />
</root>
</configuration>
Properties
logging.file.name=myapplication.log
Yaml
logging:
file:
name: "myapplication.log"
Spring Boot supports Log4j 2 for logging configuration if it is on the classpath. If you use the starters
for assembling dependencies, you have to exclude Logback and then include Log4j 2 instead. If you
do not use the starters, you need to provide (at least) spring-jcl in addition to Log4j 2.
The recommended path is through the starters, even though it requires some jiggling. The following
example shows how to set up the starters in Maven:
614
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-logging</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-log4j2</artifactId>
</dependency>
Gradle provides a few different ways to set up the starters. One way is to use a module
replacement. To do so, declare a dependency on the Log4j 2 starter and tell Gradle that any
occurrences of the default logging starter should be replaced by the Log4j 2 starter, as shown in the
following example:
dependencies {
implementation "org.springframework.boot:spring-boot-starter-log4j2"
modules {
module("org.springframework.boot:spring-boot-starter-logging") {
replacedBy("org.springframework.boot:spring-boot-starter-log4j2", "Use
Log4j2 instead of Logback")
}
}
}
The Log4j starters gather together the dependencies for common logging
NOTE requirements (such as having Tomcat use java.util.logging but configuring the
output using Log4j 2).
In addition to its default XML configuration format, Log4j 2 also supports YAML and JSON
configuration files. To configure Log4j 2 to use an alternative configuration file format, add the
appropriate dependencies to the classpath and name your configuration files to match your chosen
615
file format, as shown in the following example:
Log4j 2 has support for combining multiple configuration files into a single composite
configuration. To use this support in Spring Boot, configure logging.log4j2.config.override with
the locations of one or more secondary configuration files. The secondary configuration files will be
merged with the primary configuration, whether the primary’s source is Spring Boot’s defaults, a
standard location such as log4j.xml, or the location configured by the logging.config property.
To configure your own DataSource, define a @Bean of that type in your configuration. Spring Boot
reuses your DataSource anywhere one is required, including database initialization. If you need to
externalize some settings, you can bind your DataSource to the environment (see “Third-party
Configuration”).
Java
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "app.datasource")
public SomeDataSource dataSource() {
return new SomeDataSource();
}
616
Kotlin
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "app.datasource")
fun dataSource(): SomeDataSource {
return SomeDataSource()
}
The following example shows how to define a data source by setting properties:
Properties
app.datasource.url=jdbc:h2:mem:mydb
app.datasource.username=sa
app.datasource.pool-size=30
Yaml
app:
datasource:
url: "jdbc:h2:mem:mydb"
username: "sa"
pool-size: 30
Assuming that SomeDataSource has regular JavaBean properties for the URL, the username, and the
pool size, these settings are bound automatically before the DataSource is made available to other
components.
Spring Boot also provides a utility builder class, called DataSourceBuilder, that can be used to create
one of the standard data sources (if it is on the classpath). The builder can detect the one to use
based on what is available on the classpath. It also auto-detects the driver based on the JDBC URL.
The following example shows how to create a data source by using a DataSourceBuilder:
617
Java
import javax.sql.DataSource;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties("app.datasource")
public DataSource dataSource() {
return DataSourceBuilder.create().build();
}
Kotlin
import javax.sql.DataSource
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.jdbc.DataSourceBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties("app.datasource")
fun dataSource(): DataSource {
return DataSourceBuilder.create().build()
}
To run an app with that DataSource, all you need is the connection information. Pool-specific
settings can also be provided. Check the implementation that is going to be used at runtime for
more details.
The following example shows how to define a JDBC data source by setting properties:
618
Properties
app.datasource.url=jdbc:mysql://localhost/test
app.datasource.username=dbuser
app.datasource.password=dbpass
app.datasource.pool-size=30
Yaml
app:
datasource:
url: "jdbc:mysql://localhost/test"
username: "dbuser"
password: "dbpass"
pool-size: 30
However, there is a catch. Because the actual type of the connection pool is not exposed, no keys are
generated in the metadata for your custom DataSource and no completion is available in your IDE
(because the DataSource interface exposes no properties). Also, if you happen to have Hikari on the
classpath, this basic setup does not work, because Hikari has no url property (but does have a
jdbcUrl property). In that case, you must rewrite your configuration as follows:
Properties
app.datasource.jdbc-url=jdbc:mysql://localhost/test
app.datasource.username=dbuser
app.datasource.password=dbpass
app.datasource.pool-size=30
Yaml
app:
datasource:
jdbc-url: "jdbc:mysql://localhost/test"
username: "dbuser"
password: "dbpass"
pool-size: 30
You can fix that by forcing the connection pool to use and return a dedicated implementation
rather than DataSource. You cannot change the implementation at runtime, but the list of options
will be explicit.
619
Java
import com.zaxxer.hikari.HikariDataSource;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties("app.datasource")
public HikariDataSource dataSource() {
return DataSourceBuilder.create().type(HikariDataSource.class).build();
}
Kotlin
import com.zaxxer.hikari.HikariDataSource
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.jdbc.DataSourceBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyDataSourceConfiguration {
@Bean
@ConfigurationProperties("app.datasource")
fun dataSource(): HikariDataSource {
return DataSourceBuilder.create().type(HikariDataSource::class.java).build()
}
You can even go further by leveraging what DataSourceProperties does for you — that is, by
providing a default embedded database with a sensible username and password if no URL is
provided. You can easily initialize a DataSourceBuilder from the state of any DataSourceProperties
object, so you could also inject the DataSource that Spring Boot creates automatically. However, that
would split your configuration into two namespaces: url, username, password, type, and driver on
spring.datasource and the rest on your custom namespace (app.datasource). To avoid that, you can
redefine a custom DataSourceProperties on your custom namespace, as shown in the following
example:
620
Java
import com.zaxxer.hikari.HikariDataSource;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
@Configuration(proxyBeanMethods = false)
public class MyDataSourceConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource")
public DataSourceProperties dataSourceProperties() {
return new DataSourceProperties();
}
@Bean
@ConfigurationProperties("app.datasource.configuration")
public HikariDataSource dataSource(DataSourceProperties properties) {
return
properties.initializeDataSourceBuilder().type(HikariDataSource.class).build();
}
621
Kotlin
import com.zaxxer.hikari.HikariDataSource
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Primary
@Configuration(proxyBeanMethods = false)
class MyDataSourceConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource")
fun dataSourceProperties(): DataSourceProperties {
return DataSourceProperties()
}
@Bean
@ConfigurationProperties("app.datasource.configuration")
fun dataSource(properties: DataSourceProperties): HikariDataSource {
return
properties.initializeDataSourceBuilder().type(HikariDataSource::class.java).build()
}
This setup puts you in sync with what Spring Boot does for you by default, except that a dedicated
connection pool is chosen (in code) and its settings are exposed in the app.datasource.configuration
sub namespace. Because DataSourceProperties is taking care of the url/jdbcUrl translation for you,
you can configure it as follows:
Properties
app.datasource.url=jdbc:mysql://localhost/test
app.datasource.username=dbuser
app.datasource.password=dbpass
app.datasource.configuration.maximum-pool-size=30
Yaml
app:
datasource:
url: "jdbc:mysql://localhost/test"
username: "dbuser"
password: "dbpass"
configuration:
maximum-pool-size: 30
622
Spring Boot will expose Hikari-specific settings to spring.datasource.hikari. This
TIP example uses a more generic configuration sub namespace as the example does not
support multiple datasource implementations.
See “Configure a DataSource” in the “Spring Boot features” section and the
DataSourceAutoConfiguration class for more details.
If you need to configure multiple data sources, you can apply the same tricks that are described in
the previous section. You must, however, mark one of the DataSource instances as @Primary, because
various auto-configurations down the road expect to be able to get one by type.
If you create your own DataSource, the auto-configuration backs off. In the following example, we
provide the exact same feature set as the auto-configuration provides on the primary data source:
623
Java
import com.zaxxer.hikari.HikariDataSource;
import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
@Configuration(proxyBeanMethods = false)
public class MyDataSourcesConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource.first")
public DataSourceProperties firstDataSourceProperties() {
return new DataSourceProperties();
}
@Bean
@Primary
@ConfigurationProperties("app.datasource.first.configuration")
public HikariDataSource firstDataSource(DataSourceProperties
firstDataSourceProperties) {
return
firstDataSourceProperties.initializeDataSourceBuilder().type(HikariDataSource.class).b
uild();
}
@Bean
@ConfigurationProperties("app.datasource.second")
public BasicDataSource secondDataSource() {
return DataSourceBuilder.create().type(BasicDataSource.class).build();
}
624
Kotlin
import com.zaxxer.hikari.HikariDataSource
import org.apache.commons.dbcp2.BasicDataSource
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.jdbc.DataSourceBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Primary
@Configuration(proxyBeanMethods = false)
class MyDataSourcesConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource.first")
fun firstDataSourceProperties(): DataSourceProperties {
return DataSourceProperties()
}
@Bean
@Primary
@ConfigurationProperties("app.datasource.first.configuration")
fun firstDataSource(firstDataSourceProperties: DataSourceProperties):
HikariDataSource {
return
firstDataSourceProperties.initializeDataSourceBuilder().type(HikariDataSource::class.j
ava).build()
}
@Bean
@ConfigurationProperties("app.datasource.second")
fun secondDataSource(): BasicDataSource {
return DataSourceBuilder.create().type(BasicDataSource::class.java).build()
}
Both data sources are also bound for advanced customizations. For instance, you could configure
them as follows:
625
Properties
app.datasource.first.url=jdbc:mysql://localhost/first
app.datasource.first.username=dbuser
app.datasource.first.password=dbpass
app.datasource.first.configuration.maximum-pool-size=30
app.datasource.second.url=jdbc:mysql://localhost/second
app.datasource.second.username=dbuser
app.datasource.second.password=dbpass
app.datasource.second.max-total=30
Yaml
app:
datasource:
first:
url: "jdbc:mysql://localhost/first"
username: "dbuser"
password: "dbpass"
configuration:
maximum-pool-size: 30
second:
url: "jdbc:mysql://localhost/second"
username: "dbuser"
password: "dbpass"
max-total: 30
You can apply the same concept to the secondary DataSource as well, as shown in the following
example:
626
Java
import com.zaxxer.hikari.HikariDataSource;
import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
@Configuration(proxyBeanMethods = false)
public class MyCompleteDataSourcesConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource.first")
public DataSourceProperties firstDataSourceProperties() {
return new DataSourceProperties();
}
@Bean
@Primary
@ConfigurationProperties("app.datasource.first.configuration")
public HikariDataSource firstDataSource(DataSourceProperties
firstDataSourceProperties) {
return
firstDataSourceProperties.initializeDataSourceBuilder().type(HikariDataSource.class).b
uild();
}
@Bean
@ConfigurationProperties("app.datasource.second")
public DataSourceProperties secondDataSourceProperties() {
return new DataSourceProperties();
}
@Bean
@ConfigurationProperties("app.datasource.second.configuration")
public BasicDataSource secondDataSource(
@Qualifier("secondDataSourceProperties") DataSourceProperties
secondDataSourceProperties) {
return
secondDataSourceProperties.initializeDataSourceBuilder().type(BasicDataSource.class).b
uild();
}
627
Kotlin
import com.zaxxer.hikari.HikariDataSource
import org.apache.commons.dbcp2.BasicDataSource
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Primary
@Configuration(proxyBeanMethods = false)
class MyCompleteDataSourcesConfiguration {
@Bean
@Primary
@ConfigurationProperties("app.datasource.first")
fun firstDataSourceProperties(): DataSourceProperties {
return DataSourceProperties()
}
@Bean
@Primary
@ConfigurationProperties("app.datasource.first.configuration")
fun firstDataSource(firstDataSourceProperties: DataSourceProperties):
HikariDataSource {
return
firstDataSourceProperties.initializeDataSourceBuilder().type(HikariDataSource::class.j
ava).build()
}
@Bean
@ConfigurationProperties("app.datasource.second")
fun secondDataSourceProperties(): DataSourceProperties {
return DataSourceProperties()
}
@Bean
@ConfigurationProperties("app.datasource.second.configuration")
fun secondDataSource(secondDataSourceProperties: DataSourceProperties):
BasicDataSource {
return
secondDataSourceProperties.initializeDataSourceBuilder().type(BasicDataSource::class.j
ava).build()
}
The preceding example configures two data sources on custom namespaces with the same logic as
Spring Boot would use in auto-configuration. Note that each configuration sub namespace provides
advanced settings based on the chosen implementation.
628
18.8.3. Use Spring Data Repositories
Spring Data can create implementations of @Repository interfaces of various flavors. Spring Boot
handles all of that for you, as long as those @Repositories are included in the same package (or a
sub-package) of your @EnableAutoConfiguration class.
For many applications, all you need is to put the right Spring Data dependencies on your classpath.
There is a spring-boot-starter-data-jpa for JPA, spring-boot-starter-data-mongodb for Mongodb,
and various other starters for supported technologies. To get started, create some repository
interfaces to handle your @Entity objects.
Spring Boot tries to guess the location of your @Repository definitions, based on the
@EnableAutoConfiguration it finds. To get more control, use the @EnableJpaRepositories annotation
(from Spring Data JPA).
For more about Spring Data, see the Spring Data project page.
Spring Boot tries to guess the location of your @Entity definitions, based on the
@EnableAutoConfiguration it finds. To get more control, you can use the @EntityScan annotation, as
shown in the following example:
Java
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.domain.EntityScan;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
@EnableAutoConfiguration
@EntityScan(basePackageClasses = City.class)
public class MyApplication {
// ...
629
Kotlin
import org.springframework.boot.autoconfigure.EnableAutoConfiguration
import org.springframework.boot.autoconfigure.domain.EntityScan
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
@EnableAutoConfiguration
@EntityScan(basePackageClasses = [City::class])
class MyApplication {
// ...
Spring Data JPA already provides some vendor-independent configuration options (such as those
for SQL logging), and Spring Boot exposes those options and a few more for Hibernate as external
configuration properties. Some of them are automatically detected according to the context so you
should not have to set them.
The dialect to use is detected by the JPA provider. If you prefer to set the dialect yourself, set the
spring.jpa.database-platform property.
The most common options to set are shown in the following example:
Properties
spring.jpa.hibernate.naming.physical-strategy=com.example.MyPhysicalNamingStrategy
spring.jpa.show-sql=true
Yaml
spring:
jpa:
hibernate:
naming:
physical-strategy: "com.example.MyPhysicalNamingStrategy"
show-sql: true
In addition, all properties in spring.jpa.properties.* are passed through as normal JPA properties
(with the prefix stripped) when the local EntityManagerFactory is created.
630
You need to ensure that names defined under spring.jpa.properties.* exactly
match those expected by your JPA provider. Spring Boot will not attempt any
kind of relaxed binding for these entries.
WARNING
For example, if you want to configure Hibernate’s batch size you must use
spring.jpa.properties.hibernate.jdbc.batch_size. If you use other forms, such
as batchSize or batch-size, Hibernate will not apply the setting.
Hibernate uses two different naming strategies to map names from the object model to the
corresponding database names. The fully qualified class name of the physical and the implicit
strategy implementations can be configured by setting the spring.jpa.hibernate.naming.physical-
strategy and spring.jpa.hibernate.naming.implicit-strategy properties, respectively. Alternatively,
if ImplicitNamingStrategy or PhysicalNamingStrategy beans are available in the application context,
Hibernate will be automatically configured to use them.
631
Java
import org.hibernate.boot.model.naming.CamelCaseToUnderscoresNamingStrategy;
import org.hibernate.engine.jdbc.env.spi.JdbcEnvironment;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyHibernateConfiguration {
@Bean
public CamelCaseToUnderscoresNamingStrategy caseSensitivePhysicalNamingStrategy()
{
return new CamelCaseToUnderscoresNamingStrategy() {
@Override
protected boolean isCaseInsensitive(JdbcEnvironment jdbcEnvironment) {
return false;
}
};
}
Kotlin
import org.hibernate.boot.model.naming.CamelCaseToUnderscoresNamingStrategy
import org.hibernate.engine.jdbc.env.spi.JdbcEnvironment
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyHibernateConfiguration {
@Bean
fun caseSensitivePhysicalNamingStrategy(): CamelCaseToUnderscoresNamingStrategy {
return object : CamelCaseToUnderscoresNamingStrategy() {
override fun isCaseInsensitive(jdbcEnvironment: JdbcEnvironment): Boolean
{
return false
}
}
}
If you prefer to use Hibernate’s default instead, set the following property:
632
spring.jpa.hibernate.naming.physical-
strategy=org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl
Java
import org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
class MyHibernateConfiguration {
@Bean
PhysicalNamingStrategyStandardImpl caseSensitivePhysicalNamingStrategy() {
return new PhysicalNamingStrategyStandardImpl();
}
Kotlin
import org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
internal class MyHibernateConfiguration {
@Bean
fun caseSensitivePhysicalNamingStrategy(): PhysicalNamingStrategyStandardImpl {
return PhysicalNamingStrategyStandardImpl()
}
Hibernate second-level cache can be configured for a range of cache providers. Rather than
configuring Hibernate to lookup the cache provider again, it is better to provide the one that is
available in the context whenever possible.
To do this with JCache, first make sure that org.hibernate.orm:hibernate-jcache is available on the
classpath. Then, add a HibernatePropertiesCustomizer bean as shown in the following example:
633
Java
import org.hibernate.cache.jcache.ConfigSettings;
import org.springframework.boot.autoconfigure.orm.jpa.HibernatePropertiesCustomizer;
import org.springframework.cache.jcache.JCacheCacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyHibernateSecondLevelCacheConfiguration {
@Bean
public HibernatePropertiesCustomizer
hibernateSecondLevelCacheCustomizer(JCacheCacheManager cacheManager) {
return (properties) -> properties.put(ConfigSettings.CACHE_MANAGER,
cacheManager.getCacheManager());
}
Kotlin
import org.hibernate.cache.jcache.ConfigSettings
import org.springframework.boot.autoconfigure.orm.jpa.HibernatePropertiesCustomizer
import org.springframework.cache.jcache.JCacheCacheManager
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyHibernateSecondLevelCacheConfiguration {
@Bean
fun hibernateSecondLevelCacheCustomizer(cacheManager: JCacheCacheManager):
HibernatePropertiesCustomizer {
return HibernatePropertiesCustomizer { properties ->
properties[ConfigSettings.CACHE_MANAGER] = cacheManager.cacheManager
}
}
This customizer will configure Hibernate to use the same CacheManager as the one that the
application uses. It is also possible to use separate CacheManager instances. For details, see the
Hibernate user guide.
By default, Spring Boot registers a BeanContainer implementation that uses the BeanFactory so that
634
converters and entity listeners can use regular dependency injection.
You can disable or tune this behavior by registering a HibernatePropertiesCustomizer that removes
or changes the hibernate.resource.beans.container property.
To take full control of the configuration of the EntityManagerFactory, you need to add a @Bean named
‘entityManagerFactory’. Spring Boot auto-configuration switches off its entity manager in the
presence of a bean of that type.
If you need to use JPA against multiple data sources, you likely need one EntityManagerFactory per
data source. The LocalContainerEntityManagerFactoryBean from Spring ORM allows you to configure
an EntityManagerFactory for your needs. You can also reuse JpaProperties to bind settings for each
EntityManagerFactory, as shown in the following example:
635
Java
import javax.sql.DataSource;
import org.springframework.boot.autoconfigure.orm.jpa.JpaProperties;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.orm.jpa.JpaVendorAdapter;
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter;
@Configuration(proxyBeanMethods = false)
public class MyEntityManagerFactoryConfiguration {
@Bean
@ConfigurationProperties("app.jpa.first")
public JpaProperties firstJpaProperties() {
return new JpaProperties();
}
@Bean
public LocalContainerEntityManagerFactoryBean firstEntityManagerFactory(DataSource
firstDataSource,
JpaProperties firstJpaProperties) {
EntityManagerFactoryBuilder builder =
createEntityManagerFactoryBuilder(firstJpaProperties);
return
builder.dataSource(firstDataSource).packages(Order.class).persistenceUnit("firstDs").b
uild();
}
private EntityManagerFactoryBuilder
createEntityManagerFactoryBuilder(JpaProperties jpaProperties) {
JpaVendorAdapter jpaVendorAdapter = createJpaVendorAdapter(jpaProperties);
return new EntityManagerFactoryBuilder(jpaVendorAdapter,
jpaProperties.getProperties(), null);
}
636
Kotlin
import javax.sql.DataSource
import org.springframework.boot.autoconfigure.orm.jpa.JpaProperties
import org.springframework.boot.context.properties.ConfigurationProperties
import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.orm.jpa.JpaVendorAdapter
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter
@Configuration(proxyBeanMethods = false)
class MyEntityManagerFactoryConfiguration {
@Bean
@ConfigurationProperties("app.jpa.first")
fun firstJpaProperties(): JpaProperties {
return JpaProperties()
}
@Bean
fun firstEntityManagerFactory(
firstDataSource: DataSource?,
firstJpaProperties: JpaProperties
): LocalContainerEntityManagerFactoryBean {
val builder = createEntityManagerFactoryBuilder(firstJpaProperties)
return
builder.dataSource(firstDataSource).packages(Order::class.java).persistenceUnit("first
Ds").build()
}
637
additional JPA properties using the app.first.jpa namespace.
You should provide a similar configuration for any additional data sources for which you need JPA
access. To complete the picture, you need to configure a JpaTransactionManager for each
EntityManagerFactory as well. Alternatively, you might be able to use a JTA transaction manager that
spans both.
If you use Spring Data, you need to configure @EnableJpaRepositories accordingly, as shown in the
following examples:
Java
import org.springframework.context.annotation.Configuration;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
@Configuration(proxyBeanMethods = false)
@EnableJpaRepositories(basePackageClasses = Order.class, entityManagerFactoryRef =
"firstEntityManagerFactory")
public class OrderConfiguration {
Kotlin
import org.springframework.context.annotation.Configuration
import org.springframework.data.jpa.repository.config.EnableJpaRepositories
@Configuration(proxyBeanMethods = false)
@EnableJpaRepositories(basePackageClasses = [Order::class], entityManagerFactoryRef =
"firstEntityManagerFactory")
class OrderConfiguration
638
Java
import org.springframework.context.annotation.Configuration;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
@Configuration(proxyBeanMethods = false)
@EnableJpaRepositories(basePackageClasses = Customer.class, entityManagerFactoryRef =
"secondEntityManagerFactory")
public class CustomerConfiguration {
Kotlin
import org.springframework.context.annotation.Configuration
import org.springframework.data.jpa.repository.config.EnableJpaRepositories
@Configuration(proxyBeanMethods = false)
@EnableJpaRepositories(basePackageClasses = [Customer::class], entityManagerFactoryRef
= "secondEntityManagerFactory")
class CustomerConfiguration
Spring Boot will not search for or use a META-INF/persistence.xml by default. If you prefer to use a
traditional persistence.xml, you need to define your own @Bean of type
LocalEntityManagerFactoryBean (with an ID of ‘entityManagerFactory’) and set the persistence unit
name there.
Spring Data JPA and Spring Data Mongo can both automatically create Repository implementations
for you. If they are both present on the classpath, you might have to do some extra configuration to
tell Spring Boot which repositories to create. The most explicit way to do that is to use the standard
Spring Data @EnableJpaRepositories and @EnableMongoRepositories annotations and provide the
location of your Repository interfaces.
The same obstacle and the same features exist for other auto-configured Spring Data repository
types (Elasticsearch, Redis, and others). To work with them, change the names of the annotations
and flags accordingly.
639
18.8.13. Customize Spring Data’s Web Support
Spring Data provides web support that simplifies the use of Spring Data repositories in a web
application. Spring Boot provides properties in the spring.data.web namespace for customizing its
configuration. Note that if you are using Spring Data REST, you must use the properties in the
spring.data.rest namespace instead.
Spring Data REST can expose the Repository implementations as REST endpoints for you, provided
Spring MVC has been enabled for the application.
Spring Boot exposes a set of useful properties (from the spring.data.rest namespace) that
customize the RepositoryRestConfiguration. If you need to provide additional customization, you
should use a RepositoryRestConfigurer bean.
If you want to configure a component that JPA uses, then you need to ensure that the component is
initialized before JPA. When the component is auto-configured, Spring Boot takes care of this for
you. For example, when Flyway is auto-configured, Hibernate is configured to depend upon Flyway
so that Flyway has a chance to initialize the database before Hibernate tries to use it.
640
Java
import jakarta.persistence.EntityManagerFactory;
import
org.springframework.boot.autoconfigure.orm.jpa.EntityManagerFactoryDependsOnPostProces
sor;
import org.springframework.stereotype.Component;
/**
* {@link EntityManagerFactoryDependsOnPostProcessor} that ensures that
* {@link EntityManagerFactory} beans depend on the {@code elasticsearchClient} bean.
*/
@Component
public class ElasticsearchEntityManagerFactoryDependsOnPostProcessor
extends EntityManagerFactoryDependsOnPostProcessor {
public ElasticsearchEntityManagerFactoryDependsOnPostProcessor() {
super("elasticsearchClient");
}
Kotlin
import
org.springframework.boot.autoconfigure.orm.jpa.EntityManagerFactoryDependsOnPostProces
sor
import org.springframework.stereotype.Component
@Component
class ElasticsearchEntityManagerFactoryDependsOnPostProcessor :
EntityManagerFactoryDependsOnPostProcessor("elasticsearchClient")
If you need to use jOOQ with multiple data sources, you should create your own DSLContext for each
one. See JooqAutoConfiguration for more details.
641
18.9.1. Initialize a Database Using JPA
JPA has features for DDL generation, and these can be set up to run on startup against the database.
This is controlled through two external properties:
• spring.jpa.generate-ddl (boolean) switches the feature on and off and is vendor independent.
You can set spring.jpa.hibernate.ddl-auto explicitly and the standard Hibernate property values
are none, validate, update, create, and create-drop. Spring Boot chooses a default value for you
based on whether it thinks your database is embedded. It defaults to create-drop if no schema
manager has been detected or none in all other cases. An embedded database is detected by looking
at the Connection type and JDBC url. hsqldb, h2, and derby are candidates, and others are not. Be
careful when switching from in-memory to a ‘real’ database that you do not make assumptions
about the existence of the tables and data in the new platform. You either have to set ddl-auto
explicitly or use one of the other mechanisms to initialize the database.
You can output the schema creation by enabling the org.hibernate.SQL logger. This
NOTE
is done for you automatically if you enable the debug mode.
In addition, a file named import.sql in the root of the classpath is executed on startup if Hibernate
creates the schema from scratch (that is, if the ddl-auto property is set to create or create-drop).
This can be useful for demos and for testing if you are careful but is probably not something you
want to be on the classpath in production. It is a Hibernate feature (and has nothing to do with
Spring).
Spring Boot can automatically create the schema (DDL scripts) of your JDBC DataSource or R2DBC
ConnectionFactory and initialize it (DML scripts). It loads SQL from the standard root classpath
locations: schema.sql and data.sql, respectively. In addition, Spring Boot processes the schema-
${platform}.sql and data-${platform}.sql files (if present), where platform is the value of
spring.sql.init.platform. This allows you to switch to database-specific scripts if necessary. For
example, you might choose to set it to the vendor name of the database (hsqldb, h2, oracle, mysql,
postgresql, and so on). By default, SQL database initialization is only performed when using an
embedded in-memory database. To always initialize an SQL database, irrespective of its type, set
spring.sql.init.mode to always. Similarly, to disable initialization, set spring.sql.init.mode to never.
By default, Spring Boot enables the fail-fast feature of its script-based database initializer. This
means that, if the scripts cause exceptions, the application fails to start. You can tune that behavior
by setting spring.sql.init.continue-on-error.
642
the schema creation performed by Hibernate, set spring.jpa.defer-datasource-initialization to
true. This will defer data source initialization until after any EntityManagerFactory beans have been
created and initialized. schema.sql can then be used to make additions to any schema creation
performed by Hibernate and data.sql can be used to populate it.
If you are using a Higher-level Database Migration Tool, like Flyway or Liquibase, you should use
them alone to create and initialize the schema. Using the basic schema.sql and data.sql scripts
alongside Flyway or Liquibase is not recommended and support will be removed in a future
release.
If you use Spring Batch, it comes pre-packaged with SQL initialization scripts for most popular
database platforms. Spring Boot can detect your database type and execute those scripts on startup.
If you use an embedded database, this happens by default. You can also enable it for any database
type, as shown in the following example:
Properties
spring.batch.jdbc.initialize-schema=always
Yaml
spring:
batch:
jdbc:
initialize-schema: "always"
You can also switch off the initialization explicitly by setting spring.batch.jdbc.initialize-schema to
never.
Spring Boot supports two higher-level migration tools: Flyway and Liquibase.
Typically, migrations are scripts in the form V<VERSION>__<NAME>.sql (with <VERSION> an underscore-
separated version, such as ‘1’ or ‘2_1’). By default, they are in a directory called
classpath:db/migration, but you can modify that location by setting spring.flyway.locations. This is
a comma-separated list of one or more classpath: or filesystem: locations. For example, the
following configuration would search for scripts in both the default classpath location and the
/opt/migration directory:
643
Properties
spring.flyway.locations=classpath:db/migration,filesystem:/opt/migration
Yaml
spring:
flyway:
locations: "classpath:db/migration,filesystem:/opt/migration"
You can also add a special {vendor} placeholder to use vendor-specific scripts. Assume the
following:
Properties
spring.flyway.locations=classpath:db/migration/{vendor}
Yaml
spring:
flyway:
locations: "classpath:db/migration/{vendor}"
Rather than using db/migration, the preceding configuration sets the directory to use according to
the type of the database (such as db/migration/mysql for MySQL). The list of supported databases is
available in DatabaseDriver.
Migrations can also be written in Java. Flyway will be auto-configured with any beans that
implement JavaMigration.
FlywayProperties provides most of Flyway’s settings and a small set of additional properties that can
be used to disable the migrations or switch off the location checking. If you need more control over
the configuration, consider registering a FlywayConfigurationCustomizer bean.
Spring Boot calls Flyway.migrate() to perform the database migration. If you would like more
control, provide a @Bean that implements FlywayMigrationStrategy.
Flyway supports SQL and Java callbacks. To use SQL-based callbacks, place the callback scripts in
the classpath:db/migration directory. To use Java-based callbacks, create one or more beans that
implement Callback. Any such beans are automatically registered with Flyway. They can be ordered
by using @Order or by implementing Ordered. Beans that implement the deprecated FlywayCallback
interface can also be detected, however they cannot be used alongside Callback beans.
By default, Flyway autowires the (@Primary) DataSource in your context and uses that for migrations.
If you like to use a different DataSource, you can create one and mark its @Bean as @FlywayDataSource.
If you do so and want two data sources, remember to create another one and mark it as @Primary.
Alternatively, you can use Flyway’s native DataSource by setting spring.flyway.[url,user,password]
in external properties. Setting either spring.flyway.url or spring.flyway.user is sufficient to cause
644
Flyway to use its own DataSource. If any of the three properties has not been set, the value of its
equivalent spring.datasource property will be used.
You can also use Flyway to provide data for specific scenarios. For example, you can place test-
specific migrations in src/test/resources and they are run only when your application starts for
testing. Also, you can use profile-specific configuration to customize spring.flyway.locations so that
certain migrations run only when a particular profile is active. For example, in application-
dev.properties, you might specify the following setting:
Properties
spring.flyway.locations=classpath:/db/migration,classpath:/dev/db/migration
Yaml
spring:
flyway:
locations: "classpath:/db/migration,classpath:/dev/db/migration"
With that setup, migrations in dev/db/migration run only when the dev profile is active.
By default, the master change log is read from db/changelog/db.changelog-master.yaml, but you can
change the location by setting spring.liquibase.change-log. In addition to YAML, Liquibase also
supports JSON, XML, and SQL change log formats.
By default, Liquibase autowires the (@Primary) DataSource in your context and uses that for
migrations. If you need to use a different DataSource, you can create one and mark its @Bean as
@LiquibaseDataSource. If you do so and you want two data sources, remember to create another one
and mark it as @Primary. Alternatively, you can use Liquibase’s native DataSource by setting
spring.liquibase.[driver-class-name,url,user,password] in external properties. Setting either
spring.liquibase.url or spring.liquibase.user is sufficient to cause Liquibase to use its own
DataSource. If any of the three properties has not been set, the value of its equivalent
spring.datasource property will be used.
See LiquibaseProperties for details about available settings such as contexts, the default schema,
and others.
645
18.9.6. Depend Upon an Initialized Database
Spring Boot will automatically detect beans of the following types that initialize an SQL database:
• DataSourceScriptDatabaseInitializer
• EntityManagerFactory
• Flyway
• FlywayMigrationInitializer
• R2dbcScriptDatabaseInitializer
• SpringLiquibase
If you are using a third-party starter for a database initialization library, it may provide a detector
such that beans of other types are also detected automatically. To have other beans be detected,
register an implementation of DatabaseInitializerDetector in META-INF/spring.factories.
Spring Boot will automatically detect beans of the following types that depends upon database
initialization:
• DSLContext (jOOQ)
• JdbcOperations
• NamedParameterJdbcOperations
If you are using a third-party starter data access library, it may provide a detector such that beans
of other types are also detected automatically. To have other beans be detected, register an
implementation of DependsOnDatabaseInitializationDetector in META-INF/spring.factories.
Alternatively, annotate the bean’s class or its @Bean method with @DependsOnDatabaseInitialization.
18.10. NoSQL
Spring Boot offers a number of starters that support NoSQL technologies. This section answers
questions that arise from using NoSQL with Spring Boot.
646
18.10.1. Use Jedis Instead of Lettuce
By default, the Spring Boot starter (spring-boot-starter-data-redis) uses Lettuce. You need to
exclude that dependency and include the Jedis one instead. Spring Boot manages both of these
dependencies so you can switch to Jedis without specifying a version.
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
<exclusions>
<exclusion>
<groupId>io.lettuce</groupId>
<artifactId>lettuce-core</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
</dependency>
dependencies {
implementation('org.springframework.boot:spring-boot-starter-data-redis') {
exclude group: 'io.lettuce', module: 'lettuce-core'
}
implementation 'redis.clients:jedis'
// ...
}
18.11. Messaging
Spring Boot offers a number of starters to support messaging. This section answers questions that
arise from using messaging with Spring Boot.
If your JMS broker does not support transacted sessions, you have to disable the support of
transactions altogether. If you create your own JmsListenerContainerFactory, there is nothing to do,
since, by default it cannot be transacted. If you want to use the
DefaultJmsListenerContainerFactoryConfigurer to reuse Spring Boot’s default, you can disable
transacted sessions, as follows:
647
Java
import jakarta.jms.ConnectionFactory;
import
org.springframework.boot.autoconfigure.jms.DefaultJmsListenerContainerFactoryConfigure
r;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jms.config.DefaultJmsListenerContainerFactory;
@Configuration(proxyBeanMethods = false)
public class MyJmsConfiguration {
@Bean
public DefaultJmsListenerContainerFactory
jmsListenerContainerFactory(ConnectionFactory connectionFactory,
DefaultJmsListenerContainerFactoryConfigurer configurer) {
DefaultJmsListenerContainerFactory listenerFactory = new
DefaultJmsListenerContainerFactory();
configurer.configure(listenerFactory, connectionFactory);
listenerFactory.setTransactionManager(null);
listenerFactory.setSessionTransacted(false);
return listenerFactory;
}
648
Kotlin
import jakarta.jms.ConnectionFactory
import
org.springframework.boot.autoconfigure.jms.DefaultJmsListenerContainerFactoryConfigure
r
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.jms.config.DefaultJmsListenerContainerFactory
@Configuration(proxyBeanMethods = false)
class MyJmsConfiguration {
@Bean
fun jmsListenerContainerFactory(connectionFactory: ConnectionFactory?,
configurer: DefaultJmsListenerContainerFactoryConfigurer):
DefaultJmsListenerContainerFactory {
val listenerFactory = DefaultJmsListenerContainerFactory()
configurer.configure(listenerFactory, connectionFactory)
listenerFactory.setTransactionManager(null)
listenerFactory.setSessionTransacted(false)
return listenerFactory
}
The preceding example overrides the default factory, and it should be applied to any other factory
that your application defines, if any.
By default, batch applications require a DataSource to store job details. Spring Batch expects a single
DataSource by default. To have it use a DataSource other than the application’s main DataSource,
declare a DataSource bean, annotating its @Bean method with @BatchDataSource. If you do so and want
two data sources, remember to mark the other one @Primary. To take greater control, add
@EnableBatchProcessing to one of your @Configuration classes or extend DefaultBatchConfiguration.
See the Javadoc of @EnableBatchProcessing and DefaultBatchConfiguration for more details.
For more info about Spring Batch, see the Spring Batch project page.
649
If a single Job is found in the application context, it is executed on startup (see
JobLauncherApplicationRunner for details). If multiple Job beans are found, the job that should be
executed must be specified using spring.batch.job.name.
To disable running a Job found in the application context, set the spring.batch.job.enabled to false.
Spring Boot converts any command line argument starting with -- to a property to add to the
Environment, see accessing command line properties. This should not be used to pass arguments to
batch jobs. To specify batch arguments on the command line, use the regular format (that is
without --), as shown in the following example:
If you specify a property of the Environment on the command line, it is ignored by the job. Consider
the following command:
Spring Batch requires a data store for the Job repository. If you use Spring Boot, you must use an
actual database. Note that it can be an in-memory database, see Configuring a Job Repository.
18.13. Actuator
Spring Boot includes the Spring Boot Actuator. This section answers questions that often arise from
its use.
In a standalone application, the Actuator HTTP port defaults to the same as the main HTTP port. To
make the application listen on a different port, set the external property: management.server.port. To
listen on a completely different network address (such as when you have an internal network for
management and an external one for user applications), you can also set management.server.address
to a valid IP address to which the server is able to bind.
For more detail, see the ManagementServerProperties source code and “Customizing the Management
Server Port” in the “Production-ready features” section.
650
18.13.2. Customize the ‘whitelabel’ Error Page
Spring Boot installs a ‘whitelabel’ error page that you see in a browser client if you encounter a
server error (machine clients consuming JSON and other media types should see a sensible
response with the right error code).
Overriding the error page with your own depends on the templating technology that you use. For
example, if you use Thymeleaf, you can add an error.html template. If you use FreeMarker, you can
add an error.ftlh template. In general, you need a View that resolves with a name of error or a
@Controller that handles the /error path. Unless you replaced some of the default configuration,
you should find a BeanNameViewResolver in your ApplicationContext, so a @Bean named error would
be one way of doing that. See ErrorMvcAutoConfiguration for more options.
See also the section on “Error Handling” for details of how to register handlers in the servlet
container.
Information returned by the /env, /configprops and /quartz endpoints can be somewhat sensitive.
All values are sanitized by default (that is replaced by ******). Viewing original values in the
unsanitized form can be configured per endpoint using the showValues property for that endpoint.
This property can be configured to have the following values:
• ALWAYS - all values are shown in their unsanitized form to all users
• WHEN_AUTHORIZED - all values are shown in their unsanitized form to authorized users
For HTTP endpoints, a user is considered to be authorized if they have authenticated and have the
roles configured by the endpoint’s roles property. By default, any authenticated user is authorized.
For JMX endpoints, all users are always authorized.
Properties
management.endpoint.env.show-values=WHEN_AUTHORIZED
management.endpoint.env.roles=admin
Yaml
management:
endpoint:
env:
show-values: WHEN_AUTHORIZED
roles: "admin"
651
The configuration above enables the ability for all users with the admin role to view all values in
their original form from the /env endpoint.
Customizing Sanitization
To take control over the sanitization, define a SanitizingFunction bean. The SanitizableData with
which the function is called provides access to the key and value as well as the PropertySource from
which they came. This allows you to, for example, sanitize every value that comes from a particular
property source. Each SanitizingFunction is called in order until a function changes the value of the
sanitizable data.
Spring Boot health indicators return a Status type to indicate the overall system health. If you want
to monitor or alert on levels of health for a particular application, you can export these statuses as
metrics with Micrometer. By default, the status codes “UP”, “DOWN”, “OUT_OF_SERVICE” and
“UNKNOWN” are used by Spring Boot. To export these, you will need to convert these states to some
set of numbers so that they can be used with a Micrometer Gauge.
652
Java
import io.micrometer.core.instrument.Gauge;
import io.micrometer.core.instrument.MeterRegistry;
import org.springframework.boot.actuate.health.HealthEndpoint;
import org.springframework.boot.actuate.health.Status;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyHealthMetricsExportConfiguration {
653
Kotlin
import io.micrometer.core.instrument.Gauge
import io.micrometer.core.instrument.MeterRegistry
import org.springframework.boot.actuate.health.HealthEndpoint
import org.springframework.boot.actuate.health.Status
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyHealthMetricsExportConfiguration(registry: MeterRegistry, healthEndpoint:
HealthEndpoint) {
init {
// This example presumes common tags (such as the app) are applied elsewhere
Gauge.builder("health", healthEndpoint) { health ->
getStatusCode(health).toDouble()
}.strongReference(true).register(registry)
}
18.14. Security
This section addresses questions about security when working with Spring Boot, including
questions that arise from using Spring Security with Spring Boot.
For more about Spring Security, see the Spring Security project page.
If you define a @Configuration with a SecurityFilterChain bean in your application, it switches off
the default webapp security settings in Spring Boot.
654
18.14.2. Change the UserDetailsService and Add User Accounts
The easiest way to add user accounts is to provide your own UserDetailsService bean.
Ensuring that all your main endpoints are only available over HTTPS is an important chore for any
application. If you use Tomcat as a servlet container, then Spring Boot adds Tomcat’s own
RemoteIpValve automatically if it detects some environment settings, and you should be able to rely
on the HttpServletRequest to report whether it is secure or not (even downstream of a proxy server
that handles the real SSL termination). The standard behavior is determined by the presence or
absence of certain request headers (x-forwarded-for and x-forwarded-proto), whose names are
conventional, so it should work with most front-end proxies. You can switch on the valve by adding
some entries to application.properties, as shown in the following example:
Properties
server.tomcat.remoteip.remote-ip-header=x-forwarded-for
server.tomcat.remoteip.protocol-header=x-forwarded-proto
Yaml
server:
tomcat:
remoteip:
remote-ip-header: "x-forwarded-for"
protocol-header: "x-forwarded-proto"
(The presence of either of those properties switches on the valve. Alternatively, you can add the
RemoteIpValve by customizing the TomcatServletWebServerFactory using a WebServerFactoryCustomizer
bean.)
To configure Spring Security to require a secure channel for all (or some) requests, consider adding
your own SecurityFilterChain bean that adds the following HttpSecurity configuration:
655
Java
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration
public class MySecurityConfig {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
// Customize the application security ...
http.requiresChannel((channel) -> channel.anyRequest().requiresSecure());
return http.build();
}
Kotlin
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.security.config.annotation.web.builders.HttpSecurity
import org.springframework.security.web.SecurityFilterChain
@Configuration
class MySecurityConfig {
@Bean
fun securityFilterChain(http: HttpSecurity): SecurityFilterChain {
// Customize the application security ...
http.requiresChannel { requests -> requests.anyRequest().requiresSecure() }
return http.build()
}
There are several options for hot reloading. The recommended approach is to use spring-boot-
devtools, as it provides additional development-time features, such as support for fast application
restarts and LiveReload as well as sensible development-time configuration (such as template
caching). Devtools works by monitoring the classpath for changes. This means that static resource
656
changes must be "built" for the change to take effect. By default, this happens automatically in
Eclipse when you save your changes. In IntelliJ IDEA, the Make Project command triggers the
necessary build. Due to the default restart exclusions, changes to static resources do not trigger a
restart of your application. They do, however, trigger a live reload.
Alternatively, running in an IDE (especially with debugging on) is a good way to do development
(all modern IDEs allow reloading of static resources and usually also allow hot-swapping of Java
class changes).
Finally, the Maven and Gradle plugins can be configured (see the addResources property) to support
running from the command line with reloading of static files directly from source. You can use that
with an external css/js compiler process if you are writing that code with higher-level tools.
Most of the templating technologies supported by Spring Boot include a configuration option to
disable caching (described later in this document). If you use the spring-boot-devtools module,
these properties are automatically configured for you at development time.
Thymeleaf Templates
If you use Thymeleaf, set spring.thymeleaf.cache to false. See ThymeleafAutoConfiguration for other
Thymeleaf customization options.
FreeMarker Templates
Groovy Templates
The spring-boot-devtools module includes support for automatic application restarts. While not as
fast as technologies such as JRebel it is usually significantly faster than a “cold start”. You should
probably give it a try before investigating some of the more complex reload options discussed later
in this document.
Many modern IDEs (Eclipse, IDEA, and others) support hot swapping of bytecode. Consequently, if
you make a change that does not affect class or method signatures, it should reload cleanly with no
side effects.
657
18.16. Testing
Spring Boot includes a number of testing utilities and support classes as well as a dedicated starter
that provides common test dependencies. This section answers common questions about testing.
Spring Security provides support for running tests as a specific user. For example, the test in the
snippet below will run with an authenticated user that has the ADMIN role.
Java
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.security.test.context.support.WithMockUser;
import org.springframework.test.web.servlet.MockMvc;
@WebMvcTest(UserController.class)
class MySecurityTests {
@Autowired
private MockMvc mvc;
@Test
@WithMockUser(roles = "ADMIN")
void requestProtectedUrlWithUser() throws Exception {
this.mvc.perform(get("/"));
}
658
Kotlin
import org.junit.jupiter.api.Test
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
import org.springframework.security.test.context.support.WithMockUser
import org.springframework.test.web.servlet.MockMvc
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders
@WebMvcTest(UserController::class)
class MySecurityTests(@Autowired val mvc: MockMvc) {
@Test
@WithMockUser(roles = ["ADMIN"])
fun requestProtectedUrlWithUser() {
mvc.perform(MockMvcRequestBuilders.get("/"))
}
Spring Security provides comprehensive integration with Spring MVC Test and this can also be used
when testing controllers using the @WebMvcTest slice and MockMvc.
For additional details on Spring Security’s testing support, see Spring Security’s reference
documentation.
Slice tests work by restricting Spring Framework’s component scanning to a limited set of
components based on their type. For any beans that are not created through component scanning,
for example, beans that are created using the @Bean annotation, slice tests will not be able to
include/exclude them from the application context. Consider this example:
659
import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MyConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.authorizeHttpRequests((requests) ->
requests.anyRequest().authenticated());
return http.build();
}
@Bean
@ConfigurationProperties("app.datasource.second")
public BasicDataSource secondDataSource() {
return DataSourceBuilder.create().type(BasicDataSource.class).build();
}
For a @WebMvcTest for an application with the above @Configuration class, you might expect to have
the SecurityFilterChain bean in the application context so that you can test if your controller
endpoints are secured properly. However, MyConfiguration is not picked up by @WebMvcTest’s
component scanning filter because it doesn’t match any of the types specified by the filter. You can
include the configuration explicitly by annotating the test class with
@Import(MyConfiguration.class). This will load all the beans in MyConfiguration including the
BasicDataSource bean which isn’t required when testing the web tier. Splitting the configuration
class into two will enable importing just the security configuration.
660
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.web.SecurityFilterChain;
@Configuration(proxyBeanMethods = false)
public class MySecurityConfiguration {
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception
{
http.authorizeHttpRequests((requests) ->
requests.anyRequest().authenticated());
return http.build();
}
import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class MyDatasourceConfiguration {
@Bean
@ConfigurationProperties("app.datasource.second")
public BasicDataSource secondDataSource() {
return DataSourceBuilder.create().type(BasicDataSource.class).build();
}
Having a single configuration class can be inefficient when beans of a certain domain need to be
included in slice tests. Instead, structuring the application’s configuration as multiple granular
classes with beans for a specific domain can enable importing them only for specific slice tests.
18.17. Build
Spring Boot includes build plugins for Maven and Gradle. This section answers common questions
about these plugins.
661
18.17.1. Generate Build Information
Both the Maven plugin and the Gradle plugin allow generating build information containing the
coordinates, name, and version of the project. The plugins can also be configured to add additional
properties through configuration. When such a file is present, Spring Boot auto-configures a
BuildProperties bean.
To generate build information with Maven, add an execution for the build-info goal, as shown in
the following example:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>3.1.0-RC2</version>
<executions>
<execution>
<goals>
<goal>build-info</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
TIP See the Spring Boot Maven Plugin documentation for more details.
springBoot {
buildInfo()
}
TIP See the Spring Boot Gradle Plugin documentation for more details.
Both Maven and Gradle allow generating a git.properties file containing information about the
state of your git source code repository when the project was built.
For Maven users, the spring-boot-starter-parent POM includes a pre-configured plugin to generate
a git.properties file. To use it, add the following declaration for the Git Commit Id Plugin to your
POM:
662
<build>
<plugins>
<plugin>
<groupId>io.github.git-commit-id</groupId>
<artifactId>git-commit-id-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
Gradle users can achieve the same result by using the gradle-git-properties plugin, as shown in the
following example:
plugins {
id "com.gorylenko.gradle-git-properties" version "2.4.1"
}
Both the Maven and Gradle plugins allow the properties that are included in git.properties to be
configured.
The commit time in git.properties is expected to match the following format: yyyy-MM-
dd’T’HH:mm:ssZ. This is the default format for both plugins listed above. Using this
TIP
format lets the time be parsed into a Date and its format, when serialized to JSON, to
be controlled by Jackson’s date serialization configuration settings.
The spring-boot-dependencies POM manages the versions of common dependencies. The Spring
Boot plugins for Maven and Gradle allow these managed dependency versions to be customized
using build properties.
Each Spring Boot release is designed and tested against this specific set of
WARNING
third-party dependencies. Overriding versions may cause compatibility issues.
To override dependency versions with Maven, see this section of the Maven plugin’s
documentation.
To override dependency versions in Gradle, see this section of the Gradle plugin’s documentation.
The spring-boot-maven-plugin can be used to create an executable “fat” JAR. If you use the spring-
boot-starter-parent POM, you can declare the plugin and your jars are repackaged as follows:
663
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
If you do not use the parent POM, you can still use the plugin. However, you must additionally add
an <executions> section, as follows:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>3.1.0-RC2</version>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
Like a war file, a Spring Boot application is not intended to be used as a dependency. If your
application contains classes that you want to share with other projects, the recommended approach
is to move that code into a separate module. The separate module can then be depended upon by
your application and other projects.
If you cannot rearrange your code as recommended above, Spring Boot’s Maven and Gradle plugins
must be configured to produce a separate artifact that is suitable for use as a dependency. The
executable archive cannot be used as a dependency as the executable jar format packages
application classes in BOOT-INF/classes. This means that they cannot be found when the executable
jar is used as a dependency.
To produce the two artifacts, one that can be used as a dependency and one that is executable, a
classifier must be specified. This classifier is applied to the name of the executable archive, leaving
the default archive for use as a dependency.
664
To configure a classifier of exec in Maven, you can use the following configuration:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<classifier>exec</classifier>
</configuration>
</plugin>
</plugins>
</build>
Most nested libraries in an executable jar do not need to be unpacked in order to run. However,
certain libraries can have problems. For example, JRuby includes its own nested jar support, which
assumes that the jruby-complete.jar is always directly available as a file in its own right.
To deal with any problematic libraries, you can flag that specific nested jars should be
automatically unpacked when the executable jar first runs. Such nested jars are written beneath
the temporary directory identified by the java.io.tmpdir system property.
Care should be taken to ensure that your operating system is configured so that
WARNING it will not delete the jars that have been unpacked to the temporary directory
while the application is still running.
For example, to indicate that JRuby should be flagged for unpacking by using the Maven Plugin, you
would add the following configuration:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<requiresUnpack>
<dependency>
<groupId>org.jruby</groupId>
<artifactId>jruby-complete</artifactId>
</dependency>
</requiresUnpack>
</configuration>
</plugin>
</plugins>
</build>
665
18.17.7. Create a Non-executable JAR with Exclusions
Often, if you have an executable and a non-executable jar as two separate build products, the
executable version has additional configuration files that are not needed in a library jar. For
example, the application.yaml configuration file might be excluded from the non-executable JAR.
In Maven, the executable jar must be the main artifact and you can add a classified jar for the
library, as follows:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<artifactId>maven-jar-plugin</artifactId>
<executions>
<execution>
<id>lib</id>
<phase>package</phase>
<goals>
<goal>jar</goal>
</goals>
<configuration>
<classifier>lib</classifier>
<excludes>
<exclude>application.yaml</exclude>
</excludes>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
To attach a remote debugger to a Spring Boot application that was started with Maven, you can use
the jvmArguments property of the maven plugin.
To build with Ant, you need to grab dependencies, compile, and then create a jar or war archive. To
make it executable, you can either use the spring-boot-antlib module or you can follow these
instructions:
666
1. If you are building a jar, package the application’s classes and resources in a nested BOOT-
INF/classes directory. If you are building a war, package the application’s classes in a nested
WEB-INF/classes directory as usual.
2. Add the runtime dependencies in a nested BOOT-INF/lib directory for a jar or WEB-INF/lib for a
war. Remember not to compress the entries in the archive.
3. Add the provided (embedded container) dependencies in a nested BOOT-INF/lib directory for a
jar or WEB-INF/lib-provided for a war. Remember not to compress the entries in the archive.
4. Add the spring-boot-loader classes at the root of the archive (so that the Main-Class is available).
5. Use the appropriate launcher (such as JarLauncher for a jar file) as a Main-Class attribute in the
manifest and specify the other properties it needs as manifest entries — principally, by setting a
Start-Class property.
The following example shows how to build an executable archive with Ant:
667
18.18.1. Create a Deployable War File
Because Spring WebFlux does not strictly depend on the servlet API and
WARNING applications are deployed by default on an embedded Reactor Netty server,
War deployment is not supported for WebFlux applications.
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
@SpringBootApplication
public class MyApplication extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application)
{
return application.sources(MyApplication.class);
}
668
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.builder.SpringApplicationBuilder
import org.springframework.boot.runApplication
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer
@SpringBootApplication
class MyApplication : SpringBootServletInitializer() {
The next step is to update your build configuration such that your project produces a war file
rather than a jar file. If you use Maven and spring-boot-starter-parent (which configures Maven’s
war plugin for you), all you need to do is to modify pom.xml to change the packaging to war, as
follows:
<packaging>war</packaging>
If you use Gradle, you need to modify build.gradle to apply the war plugin to the project, as follows:
The final step in the process is to ensure that the embedded servlet container does not interfere
with the servlet container to which the war file is deployed. To do so, you need to mark the
embedded servlet container dependency as being provided.
If you use Maven, the following example marks the servlet container (Tomcat, in this case) as being
provided:
669
<dependencies>
<!-- ... -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
<scope>provided</scope>
</dependency>
<!-- ... -->
</dependencies>
If you use Gradle, the following example marks the servlet container (Tomcat, in this case) as being
provided:
dependencies {
// ...
providedRuntime 'org.springframework.boot:spring-boot-starter-tomcat'
// ...
}
If you use the Spring Boot build tools, marking the embedded servlet container dependency as
provided produces an executable war file with the provided dependencies packaged in a lib-
provided directory. This means that, in addition to being deployable to a servlet container, you can
also run your application by using java -jar on the command line.
To convert an existing non-web Spring application to a Spring Boot application, replace the code
that creates your ApplicationContext and replace it with calls to SpringApplication or
SpringApplicationBuilder. Spring MVC web applications are generally amenable to first creating a
deployable war application and then migrating it later to an executable war or jar. See the Getting
Started Guide on Converting a jar to a war.
670
Java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
@SpringBootApplication
public class MyApplication extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application)
{
// Customize the application or call application.sources(...) to add sources
// Since our example is itself a @Configuration class (through
// @SpringBootApplication)
// we actually do not need to override this method.
return application;
}
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.builder.SpringApplicationBuilder
import org.springframework.boot.runApplication
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer
@SpringBootApplication
class MyApplication : SpringBootServletInitializer() {
Remember that, whatever you put in the sources is merely a Spring ApplicationContext. Normally,
anything that already works should work here. There might be some beans you can remove later
and let Spring Boot provide its own defaults for them, but it should be possible to get something
working before you need to do that.
Static resources can be moved to /public (or /static or /resources or /META-INF/resources) in the
671
classpath root. The same applies to messages.properties (which Spring Boot automatically detects in
the root of the classpath).
Vanilla usage of Spring DispatcherServlet and Spring Security should require no further changes. If
you have other features in your application (for instance, using other servlets or filters), you may
need to add some configuration to your Application context, by replacing those elements from the
web.xml, as follows:
• A @Bean of type Filter or FilterRegistrationBean behaves similarly (as a <filter/> and <filter-
mapping/>).
Once the war file is working, you can make it executable by adding a main method to your
Application, as shown in the following example:
Java
Kotlin
NOTE
672
Java
import org.springframework.boot.Banner;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import
org.springframework.boot.web.servlet.support.SpringBootServletInitialize
r;
@SpringBootApplication
public class MyApplication extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder
configure(SpringApplicationBuilder builder) {
return customizerBuilder(builder);
}
673
Kotlin
import org.springframework.boot.Banner
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.builder.SpringApplicationBuilder
import
org.springframework.boot.web.servlet.support.SpringBootServletInitialize
r
@SpringBootApplication
class MyApplication : SpringBootServletInitializer() {
companion object {
@JvmStatic
fun main(args: Array<String>) {
customizerBuilder(SpringApplicationBuilder()).run(*args)
}
All of these should be amenable to translation, but each might require slightly different techniques.
Servlet 3.0+ applications might translate pretty easily if they already use the Spring Servlet 3.0+
initializer support classes. Normally, all the code from an existing WebApplicationInitializer can be
moved into a SpringBootServletInitializer. If your existing application has more than one
ApplicationContext (for example, if it uses AbstractDispatcherServletInitializer) then you might be
able to combine all your context sources into a single SpringApplication. The main complication you
might encounter is if combining does not work and you need to maintain the context hierarchy. See
674
the entry on building a hierarchy for examples. An existing parent context that contains web-
specific features usually needs to be broken up so that all the ServletContextAware components are
in the child context.
Applications that are not already Spring applications might be convertible to Spring Boot
applications, and the previously mentioned guidance may help. However, you may yet encounter
problems. In that case, we suggest asking questions on Stack Overflow with a tag of spring-boot.
To deploy a Spring Boot application to WebLogic, you must ensure that your servlet initializer
directly implements WebApplicationInitializer (even if you extend from a base class that already
implements it).
Java
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
import org.springframework.web.WebApplicationInitializer;
@SpringBootApplication
public class MyApplication extends SpringBootServletInitializer implements
WebApplicationInitializer {
Kotlin
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer
import org.springframework.web.WebApplicationInitializer
@SpringBootApplication
class MyApplication : SpringBootServletInitializer(), WebApplicationInitializer
If you use Logback, you also need to tell WebLogic to prefer the packaged version rather than the
version that was pre-installed with the server. You can do so by adding a WEB-INF/weblogic.xml file
with the following contents:
675
<?xml version="1.0" encoding="UTF-8"?>
<wls:weblogic-web-app
xmlns:wls="http://xmlns.oracle.com/weblogic/weblogic-web-app"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
https://java.sun.com/xml/ns/javaee/ejb-jar_3_0.xsd
http://xmlns.oracle.com/weblogic/weblogic-web-app
https://xmlns.oracle.com/weblogic/weblogic-web-app/1.4/weblogic-web-app.xsd">
<wls:container-descriptor>
<wls:prefer-application-packages>
<wls:package-name>org.slf4j</wls:package-name>
</wls:prefer-application-packages>
</wls:container-descriptor>
</wls:weblogic-web-app>
676
Appendices
Appendix A: Common Application Properties
Various properties can be specified inside your application.properties file, inside your
application.yaml file, or as command line switches. This appendix provides a list of common Spring
Boot properties and references to the underlying classes that consume them.
Spring Boot provides various conversion mechanism with advanced value formatting,
TIP
make sure to review the properties conversion section.
Property contributions can come from additional jar files on your classpath, so you
NOTE should not consider this an exhaustive list. Also, you can define your own
properties.
677
Name Description Default Value
678
Name Description Default Value
679
Name Description Default Value
680
Name Description Default Value
681
Name Description Default Value
spring.pid.fail-on-write-error Fails if
ApplicationPidFileWriter is
used but it cannot write the
PID file.
682
Name Description Default Value
683
Name Description Default Value
684
Name Description Default Value
685
Name Description Default Value
686
Name Description Default Value
687
Name Description Default Value
688
Name Description Default Value
689
Name Description Default Value
690
Name Description Default Value
691
Name Description Default Value
692
Name Description Default Value
693
Name Description Default Value
694
Name Description Default Value
695
Name Description Default Value
696
Name Description Default Value
697
Name Description Default Value
698
Name Description Default Value
699
Name Description Default Value
701
Name Description Default Value
703
Name Description Default Value
705
Name Description Default Value
707
Name Description Default Value
708
Name Description Default Value
709
Name Description Default Value
710
Name Description Default Value
711
Name Description Default Value
712
Name Description Default Value
spring.jta.atomikos.connectionfactory Vendor-specific
.xa-connection-factory-class-name implementation of
XAConnectionFactory.
spring.jta.atomikos.connectionfactory Vendor-specific XA
.xa-properties properties.
713
Name Description Default Value
spring.jta.atomikos.datasource.xa- Vendor-specific
data-source-class-name implementation of
XAConnectionFactory.
spring.jta.atomikos.datasource.xa- Vendor-specific XA
properties properties.
714
Name Description Default Value
715
Name Description Default Value
716
Name Description Default Value
717
Name Description Default Value
718
Name Description Default Value
719
Name Description Default Value
720
Name Description Default Value
721
Name Description Default Value
722
Name Description Default Value
723
Name Description Default Value
724
Name Description Default Value
725
Name Description Default Value
726
Name Description Default Value
727
Name Description Default Value
728
Name Description Default Value
729
Name Description Default Value
730
Name Description Default Value
731
Name Description Default Value
732
Name Description Default Value
733
Name Description Default Value
spring.kafka.producer.acks Number of
acknowledgments the
producer requires the leader
to have received before
considering a request
complete.
734
Name Description Default Value
735
Name Description Default Value
736
Name Description Default Value
737
Name Description Default Value
738
Name Description Default Value
739
Name Description Default Value
740
Name Description Default Value
741
Name Description Default Value
742
Name Description Default Value
743
Name Description Default Value
744
.A.9. Web Properties
745
Name Description Default Value
746
Name Description Default Value
747
Name Description Default Value
748
Name Description Default Value
749
Name Description Default Value
750
Name Description Default Value
751
Name Description Default Value
752
Name Description Default Value
753
Name Description Default Value
754
.A.10. Templating Properties
755
Name Description Default Value
756
Name Description Default Value
spring.groovy.template.configuration. See
auto-escape GroovyMarkupConfigurer
spring.groovy.template.configuration.
auto-indent
spring.groovy.template.configuration.
auto-indent-string
spring.groovy.template.configuration.
auto-new-line
spring.groovy.template.configuration.
base-template-class
spring.groovy.template.configuration.
cache-templates
spring.groovy.template.configuration.
declaration-encoding
spring.groovy.template.configuration.
expand-empty-elements
spring.groovy.template.configuration.
locale
spring.groovy.template.configuration.
new-line-string
spring.groovy.template.configuration.
resource-loader-path
spring.groovy.template.configuration.
use-double-quotes
757
Name Description Default Value
758
Name Description Default Value
759
Name Description Default Value
760
Name Description Default Value
761
Name Description Default Value
762
Name Description Default Value
763
Name Description Default Value
764
Name Description Default Value
765
Name Description Default Value
766
Name Description Default Value
767
Name Description Default Value
768
Name Description Default Value
769
Name Description Default Value
770
Name Description Default Value
771
Name Description Default Value
772
Name Description Default Value
773
Name Description Default Value
774
Name Description Default Value
775
Name Description Default Value
776
Name Description Default Value
777
.A.14. Actuator Properties
778
Name Description Default Value
779
Name Description Default Value
780
Name Description Default Value
781
Name Description Default Value
782
Name Description Default Value
783
Name Description Default Value
784
Name Description Default Value
785
Name Description Default Value
786
Name Description Default Value
787
Name Description Default Value
788
Name Description Default Value
789
Name Description Default Value
790
Name Description Default Value
791
Name Description Default Value
792
Name Description Default Value
793
Name Description Default Value
794
Name Description Default Value
795
Name Description Default Value
796
Name Description Default Value
797
Name Description Default Value
798
Name Description Default Value
799
Name Description Default Value
800
Name Description Default Value
801
Name Description Default Value
802
Name Description Default Value
803
Name Description Default Value
804
Name Description Default Value
805
Name Description Default Value
806
Name Description Default Value
807
Name Description Default Value
808
Name Description Default Value
809
Name Description Default Value
The majority of the metadata file is generated automatically at compile time by processing all items
annotated with @ConfigurationProperties. However, it is possible to write part of the metadata
manually for corner cases or more advanced use cases.
{"groups": [
{
"name": "server",
"type": "org.springframework.boot.autoconfigure.web.ServerProperties",
"sourceType": "org.springframework.boot.autoconfigure.web.ServerProperties"
},
810
{
"name": "spring.jpa.hibernate",
"type":
"org.springframework.boot.autoconfigure.orm.jpa.JpaProperties$Hibernate",
"sourceType": "org.springframework.boot.autoconfigure.orm.jpa.JpaProperties",
"sourceMethod": "getHibernate()"
}
...
],"properties": [
{
"name": "server.port",
"type": "java.lang.Integer",
"sourceType": "org.springframework.boot.autoconfigure.web.ServerProperties"
},
{
"name": "server.address",
"type": "java.net.InetAddress",
"sourceType": "org.springframework.boot.autoconfigure.web.ServerProperties"
},
{
"name": "spring.jpa.hibernate.ddl-auto",
"type": "java.lang.String",
"description": "DDL mode. This is actually a shortcut for the
\"hibernate.hbm2ddl.auto\" property.",
"sourceType":
"org.springframework.boot.autoconfigure.orm.jpa.JpaProperties$Hibernate"
}
...
],"hints": [
{
"name": "spring.jpa.hibernate.ddl-auto",
"values": [
{
"value": "none",
"description": "Disable DDL handling."
},
{
"value": "validate",
"description": "Validate the schema, make no changes to the database."
},
{
"value": "update",
"description": "Update the schema if necessary."
},
{
"value": "create",
"description": "Create the schema and destroy previous data."
},
{
"value": "create-drop",
"description": "Create and then destroy the schema at the end of the
811
session."
}
]
}
]}
Each “property” is a configuration item that the user specifies with a given value. For example,
server.port and server.address might be specified in your application.properties/application.yaml,
as follows:
Properties
server.port=9090
server.address=127.0.0.1
Yaml
server:
port: 9090
address: 127.0.0.1
The “groups” are higher level items that do not themselves specify a value but instead provide a
contextual grouping for properties. For example, the server.port and server.address properties are
part of the server group.
It is not required that every “property” has a “group”. Some properties might exist
NOTE
in their own right.
Finally, “hints” are additional information used to assist the user in configuring a given property.
For example, when a developer is configuring the spring.jpa.hibernate.ddl-auto property, a tool
can use the hints to offer some auto-completion help for the none, validate, update, create, and
create-drop values.
Group Attributes
The JSON object contained in the groups array can contain the attributes shown in the following
table:
812
Name Type Purpose
description String A short description of the group that can be displayed to users. If
no description is available, it may be omitted. It is recommended
that descriptions be short paragraphs, with the first line
providing a concise summary. The last line in the description
should end with a period (.).
sourceType String The class name of the source that contributed this group. For
example, if the group were based on a @Bean method annotated
with @ConfigurationProperties, this attribute would contain the
fully qualified name of the @Configuration class that contains the
method. If the source type is not known, the attribute may be
omitted.
sourceMethod String The full name of the method (include parenthesis and argument
types) that contributed this group (for example, the name of a
@ConfigurationProperties annotated @Bean method). If the source
method is not known, it may be omitted.
Property Attributes
The JSON object contained in the properties array can contain the attributes described in the
following table:
813
Name Type Purpose
defaultValue Object The default value, which is used if the property is not specified. If
the type of the property is an array, it can be an array of value(s).
If the default value is unknown, it may be omitted.
deprecation Deprecation Specify whether the property is deprecated. If the field is not
deprecated or if that information is not known, it may be
omitted. The next table offers more detail about the deprecation
attribute.
The JSON object contained in the deprecation attribute of each properties element can contain the
following attributes:
Prior to Spring Boot 1.3, a single deprecated boolean attribute can be used instead of
the deprecation element. This is still supported in a deprecated fashion and should
NOTE
no longer be used. If no reason and replacement are available, an empty
deprecation object should be set.
814
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.DeprecatedConfigurationProperty;
@ConfigurationProperties("my.app")
public class MyProperties {
@Deprecated
@DeprecatedConfigurationProperty(replacement = "my.app.name")
public String getTarget() {
return this.name;
}
@Deprecated
public void setTarget(String target) {
this.name = target;
}
There is no way to set a level. warning is always assumed, since code is still handling
NOTE
the property.
The preceding code makes sure that the deprecated property still works (delegating to the name
property behind the scenes). Once the getTarget and setTarget methods can be removed from your
public API, the automatic deprecation hint in the metadata goes away as well. If you want to keep a
hint, adding manual metadata with an error deprecation level ensures that users are still informed
about that property. Doing so is particularly useful when a replacement is provided.
Hint Attributes
The JSON object contained in the hints array can contain the attributes shown in the following
table:
815
Name Type Purpose
name String The full name of the property to which this hint refers. Names
are in lower-case period-separated form (such as
spring.mvc.servlet.path). If the property refers to a map (such as
system.contexts), the hint either applies to the keys of the map
(system.contexts.keys) or the values (system.contexts.values) of
the map. This attribute is mandatory.
values ValueHint[] A list of valid values as defined by the ValueHint object (described
in the next table). Each entry defines the value and may have a
description.
providers ValueProvider[ A list of providers as defined by the ValueProvider object
] (described later in this document). Each entry defines the name
of the provider and its parameters, if any.
The JSON object contained in the values attribute of each hint element can contain the attributes
described in the following table:
The JSON object contained in the providers attribute of each hint element can contain the attributes
described in the following table:
Objects with the same “property” and “group” name can appear multiple times within a metadata
file. For example, you could bind two separate classes to the same prefix, with each having
potentially overlapping property names. While the same names appearing in the metadata multiple
times should not be common, consumers of metadata should take care to ensure that they support
it.
816
.B.2. Providing Manual Hints
To improve the user experience and further assist the user in configuring a given property, you can
provide additional metadata that:
• Associates a provider, to attach a well defined semantic to a property, so that a tool can discover
the list of potential values based on the project’s context.
Value Hint
The name attribute of each hint refers to the name of a property. In the initial example shown earlier,
we provide five values for the spring.jpa.hibernate.ddl-auto property: none, validate, update,
create, and create-drop. Each value may have a description as well.
If your property is of type Map, you can provide hints for both the keys and the values (but not for
the map itself). The special .keys and .values suffixes must refer to the keys and the values,
respectively.
Assume a my.contexts maps magic String values to an integer, as shown in the following example:
import java.util.Map;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("my")
public class MyProperties {
The magic values are (in this example) are sample1 and sample2. In order to offer additional content
assistance for the keys, you could add the following JSON to the manual metadata of the module:
817
{"hints": [
{
"name": "my.contexts.keys",
"values": [
{
"value": "sample1"
},
{
"value": "sample2"
}
]
}
]}
We recommend that you use an Enum for those two values instead. If your IDE supports
TIP
it, this is by far the most effective approach to auto-completion.
Value Providers
Providers are a powerful way to attach semantics to a property. In this section, we define the
official providers that you can use for your own hints. However, your favorite IDE may implement
some of these or none of them. Also, it could eventually provide its own.
As this is a new feature, IDE vendors must catch up with how it works. Adoption
NOTE
times naturally vary.
Name Description
any Permits any additional value to be provided.
class-reference Auto-completes the classes available in the project. Usually
constrained by a base class that is specified by the target
parameter.
handle-as Handles the property as if it were defined by the type defined by
the mandatory target parameter.
logger-name Auto-completes valid logger names and logger groups. Typically,
package and class names available in the current project can be
auto-completed as well as defined groups.
spring-bean-reference Auto-completes the available bean names in the current project.
Usually constrained by a base class that is specified by the target
parameter.
spring-profile-name Auto-completes the available Spring profile names in the project.
818
Only one provider can be active for a given property, but you can specify several
providers if they can all manage the property in some way. Make sure to place the
TIP most powerful provider first, as the IDE must use the first one in the JSON section that
it can handle. If no provider for a given property is supported, no special content
assistance is provided, either.
Any
The special any provider value permits any additional values to be provided. Regular value
validation based on the property type should be applied if this is supported.
This provider is typically used if you have a list of values and any extra values should still be
considered as valid.
The following example offers on and off as auto-completion values for system.state:
{"hints": [
{
"name": "system.state",
"values": [
{
"value": "on"
},
{
"value": "off"
}
],
"providers": [
{
"name": "any"
}
]
}
]}
Note that, in the preceding example, any other value is also allowed.
Class Reference
The class-reference provider auto-completes classes available in the project. This provider
supports the following parameters:
819
Parameter Type Default value Description
target String none The fully qualified name of the class that should
(Class) be assignable to the chosen value. Typically used
to filter out-non candidate classes. Note that this
information can be provided by the type itself by
exposing a class with the appropriate upper
bound.
concrete boolean true Specify whether only concrete classes are to be
considered as valid candidates.
{"hints": [
{
"name": "server.servlet.jsp.class-name",
"providers": [
{
"name": "class-reference",
"parameters": {
"target": "jakarta.servlet.http.HttpServlet"
}
}
]
}
]}
Handle As
The handle-as provider lets you substitute the type of the property to a more high-level type. This
typically happens when the property has a java.lang.String type, because you do not want your
configuration classes to rely on classes that may not be on the classpath. This provider supports the
following parameters:
• Any java.lang.Enum: Lists the possible values for the property. (We recommend defining the
property with the Enum type, as no further hint should be required for the IDE to auto-complete
the values)
820
text/plain)
If multiple values can be provided, use a Collection or Array type to teach the IDE
TIP
about it.
{"hints": [
{
"name": "spring.liquibase.change-log",
"providers": [
{
"name": "handle-as",
"parameters": {
"target": "org.springframework.core.io.Resource"
}
}
]
}
]}
Logger Name
The logger-name provider auto-completes valid logger names and logger groups. Typically,
package and class names available in the current project can be auto-completed. If groups are
enabled (default) and if a custom logger group is identified in the configuration, auto-completion
for it should be provided. Specific frameworks may have extra magic logger names that can be
supported as well.
Since a logger name can be any arbitrary name, this provider should allow any value but could
highlight valid package and class names that are not available in the project’s classpath.
The following metadata snippet corresponds to the standard logging.level property. Keys are
logger names, and values correspond to the standard log levels or any custom level. As Spring Boot
defines a few logger groups out-of-the-box, dedicated value hints have been added for those.
821
{"hints": [
{
"name": "logging.level.keys",
"values": [
{
"value": "root",
"description": "Root logger used to assign the default logging level."
},
{
"value": "sql",
"description": "SQL logging group including Hibernate SQL logger."
},
{
"value": "web",
"description": "Web logging group including codecs."
}
],
"providers": [
{
"name": "logger-name"
}
]
},
{
"name": "logging.level.values",
"values": [
{
"value": "trace"
},
{
"value": "debug"
},
{
"value": "info"
},
{
"value": "warn"
},
{
"value": "error"
},
{
"value": "fatal"
},
{
"value": "off"
}
],
"providers": [
{
822
"name": "any"
}
]
}
]}
The spring-bean-reference provider auto-completes the beans that are defined in the
configuration of the current project. This provider supports the following parameters:
The following metadata snippet corresponds to the standard spring.jmx.server property that
defines the name of the MBeanServer bean to use:
{"hints": [
{
"name": "spring.jmx.server",
"providers": [
{
"name": "spring-bean-reference",
"parameters": {
"target": "javax.management.MBeanServer"
}
}
]
}
]}
The binder is not aware of the metadata. If you provide that hint, you still need to
NOTE transform the bean name into an actual Bean reference using by the
ApplicationContext.
The spring-profile-name provider auto-completes the Spring profiles that are defined in the
configuration of the current project.
The following metadata snippet corresponds to the standard spring.profiles.active property that
defines the name of the Spring profile(s) to enable:
823
{"hints": [
{
"name": "spring.profiles.active",
"providers": [
{
"name": "spring-profile-name"
}
]
}
]}
You can easily generate your own configuration metadata file from items annotated with
@ConfigurationProperties by using the spring-boot-configuration-processor jar. The jar includes a
Java annotation processor which is invoked as your project is compiled.
With Maven the dependency should be declared as optional, as shown in the following example:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
dependencies {
annotationProcessor "org.springframework.boot:spring-boot-configuration-processor"
}
tasks.named('compileJava') {
inputs.files(tasks.named('processResources'))
}
This dependency ensures that the additional metadata is available when the annotation processor
runs during compilation.
824
If you are using AspectJ in your project, you need to make sure that the annotation
processor runs only once. There are several ways to do this. With Maven, you can
configure the maven-apt-plugin explicitly and add the dependency to the annotation
processor only there. You could also let the AspectJ plugin run all the processing
and disable annotation processing in the maven-compiler-plugin configuration, as
follows:
NOTE
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<proc>none</proc>
</configuration>
</plugin>
If you are using Lombok in your project, you need to make sure that its annotation
processor runs before spring-boot-configuration-processor. To do so with Maven,
you can list the annotation processors in the right order using the
NOTE annotationProcessors attribute of the Maven compiler plugin. If you are not using
this attribute, and annotation processors are picked up by the dependencies
available on the classpath, make sure that the lombok dependency is defined before
the spring-boot-configuration-processor dependency.
The processor picks up both classes and methods that are annotated with @ConfigurationProperties.
If the class has a single parameterized constructor, one property is created per constructor
parameter, unless the constructor is annotated with @Autowired. If the class has a constructor
explicitly annotated with @ConstructorBinding, one property is created per constructor parameter
for that constructor. Otherwise, properties are discovered through the presence of standard getters
and setters with special handling for collection and map types (that is detected even if only a getter
is present). The annotation processor also supports the use of the @Data, @Value, @Getter, and @Setter
lombok annotations.
825
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties(prefix = "my.server")
public class MyServerProperties {
/**
* Name of the server.
*/
private String name;
/**
* IP address to listen to.
*/
private String ip = "127.0.0.1";
/**
* Port to listener to.
*/
private int port = 9797;
This exposes three properties where my.server.name has no default and my.server.ip and
my.server.port defaults to "127.0.0.1" and 9797 respectively. The Javadoc on fields is used to
826
populate the description attribute. For instance, the description of my.server.ip is "IP address to
listen to.".
You should only use plain text with @ConfigurationProperties field Javadoc, since
NOTE
they are not processed before being added to the JSON.
The annotation processor applies a number of heuristics to extract the default value from the
source model. Default values have to be provided statically. In particular, do not refer to a constant
defined in another class. Also, the annotation processor cannot auto-detect default values for Enums
and Collectionss.
For cases where the default value could not be detected, manual metadata should be provided.
Consider the following example:
827
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties(prefix = "my.messaging")
public class MyMessagingProperties {
SIMPLE, DIRECT
In order to document default values for properties in the class above, you could add the following
content to the manual metadata of the module:
828
{"properties": [
{
"name": "my.messaging.addresses",
"defaultValue": ["a", "b"]
},
{
"name": "my.messaging.container-type",
"defaultValue": "simple"
}
]}
Only the name of the property is required to document additional metadata for
NOTE
existing properties.
Nested Properties
The annotation processor automatically considers inner classes as nested properties. Rather than
documenting the ip and port at the root of the namespace, we could create a sub-namespace for it.
Consider the updated example:
829
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties(prefix = "my.server")
public class MyServerProperties {
830
The preceding example produces metadata information for my.server.name, my.server.host.ip, and
my.server.host.port properties. You can use the @NestedConfigurationProperty annotation on a field
to indicate that a regular (non-inner) class should be treated as if it were nested.
This has no effect on collections and maps, as those types are automatically identified,
TIP
and a single metadata property is generated for each of them.
Spring Boot’s configuration file handling is quite flexible, and it is often the case that properties
may exist that are not bound to a @ConfigurationProperties bean. You may also need to tune some
attributes of an existing key. To support such cases and let you provide custom "hints", the
annotation processor automatically merges items from META-INF/additional-spring-configuration-
metadata.json into the main metadata file.
If you refer to a property that has been detected automatically, the description, default value, and
deprecation information are overridden, if specified. If the manual property declaration is not
identified in the current module, it is added as a new property.
.C.1. spring-boot-autoconfigure
831
Configuration Class Links
CassandraReactiveRepositoriesAutoConfiguration javadoc
CassandraRepositoriesAutoConfiguration javadoc
ClientHttpConnectorAutoConfiguration javadoc
CodecsAutoConfiguration javadoc
ConfigurationPropertiesAutoConfiguration javadoc
CouchbaseAutoConfiguration javadoc
CouchbaseDataAutoConfiguration javadoc
CouchbaseReactiveDataAutoConfiguration javadoc
CouchbaseReactiveRepositoriesAutoConfiguration javadoc
CouchbaseRepositoriesAutoConfiguration javadoc
DataSourceAutoConfiguration javadoc
DataSourceTransactionManagerAutoConfiguration javadoc
DispatcherServletAutoConfiguration javadoc
ElasticsearchClientAutoConfiguration javadoc
ElasticsearchDataAutoConfiguration javadoc
ElasticsearchRepositoriesAutoConfiguration javadoc
ElasticsearchRestClientAutoConfiguration javadoc
EmbeddedLdapAutoConfiguration javadoc
EmbeddedWebServerFactoryCustomizerAutoConfiguration javadoc
ErrorMvcAutoConfiguration javadoc
ErrorWebFluxAutoConfiguration javadoc
FlywayAutoConfiguration javadoc
FreeMarkerAutoConfiguration javadoc
GraphQlAutoConfiguration javadoc
GraphQlQueryByExampleAutoConfiguration javadoc
GraphQlQuerydslAutoConfiguration javadoc
GraphQlRSocketAutoConfiguration javadoc
GraphQlReactiveQueryByExampleAutoConfiguration javadoc
GraphQlReactiveQuerydslAutoConfiguration javadoc
GraphQlWebFluxAutoConfiguration javadoc
GraphQlWebFluxSecurityAutoConfiguration javadoc
GraphQlWebMvcAutoConfiguration javadoc
GraphQlWebMvcSecurityAutoConfiguration javadoc
832
Configuration Class Links
GroovyTemplateAutoConfiguration javadoc
GsonAutoConfiguration javadoc
H2ConsoleAutoConfiguration javadoc
HazelcastAutoConfiguration javadoc
HazelcastJpaDependencyAutoConfiguration javadoc
HibernateJpaAutoConfiguration javadoc
HttpEncodingAutoConfiguration javadoc
HttpHandlerAutoConfiguration javadoc
HttpMessageConvertersAutoConfiguration javadoc
HypermediaAutoConfiguration javadoc
InfluxDbAutoConfiguration javadoc
IntegrationAutoConfiguration javadoc
JacksonAutoConfiguration javadoc
JdbcRepositoriesAutoConfiguration javadoc
JdbcTemplateAutoConfiguration javadoc
JerseyAutoConfiguration javadoc
JmsAutoConfiguration javadoc
JmxAutoConfiguration javadoc
JndiConnectionFactoryAutoConfiguration javadoc
JndiDataSourceAutoConfiguration javadoc
JooqAutoConfiguration javadoc
JpaRepositoriesAutoConfiguration javadoc
JsonbAutoConfiguration javadoc
JtaAutoConfiguration javadoc
KafkaAutoConfiguration javadoc
LdapAutoConfiguration javadoc
LdapRepositoriesAutoConfiguration javadoc
LifecycleAutoConfiguration javadoc
LiquibaseAutoConfiguration javadoc
MailSenderAutoConfiguration javadoc
MailSenderValidatorAutoConfiguration javadoc
MessageSourceAutoConfiguration javadoc
MongoAutoConfiguration javadoc
833
Configuration Class Links
MongoDataAutoConfiguration javadoc
MongoReactiveAutoConfiguration javadoc
MongoReactiveDataAutoConfiguration javadoc
MongoReactiveRepositoriesAutoConfiguration javadoc
MongoRepositoriesAutoConfiguration javadoc
MultipartAutoConfiguration javadoc
MustacheAutoConfiguration javadoc
Neo4jAutoConfiguration javadoc
Neo4jDataAutoConfiguration javadoc
Neo4jReactiveDataAutoConfiguration javadoc
Neo4jReactiveRepositoriesAutoConfiguration javadoc
Neo4jRepositoriesAutoConfiguration javadoc
NettyAutoConfiguration javadoc
OAuth2AuthorizationServerAutoConfiguration javadoc
OAuth2AuthorizationServerJwtAutoConfiguration javadoc
OAuth2ClientAutoConfiguration javadoc
OAuth2ResourceServerAutoConfiguration javadoc
PersistenceExceptionTranslationAutoConfiguration javadoc
ProjectInfoAutoConfiguration javadoc
PropertyPlaceholderAutoConfiguration javadoc
QuartzAutoConfiguration javadoc
R2dbcAutoConfiguration javadoc
R2dbcDataAutoConfiguration javadoc
R2dbcRepositoriesAutoConfiguration javadoc
R2dbcTransactionManagerAutoConfiguration javadoc
RSocketGraphQlClientAutoConfiguration javadoc
RSocketMessagingAutoConfiguration javadoc
RSocketRequesterAutoConfiguration javadoc
RSocketSecurityAutoConfiguration javadoc
RSocketServerAutoConfiguration javadoc
RSocketStrategiesAutoConfiguration javadoc
RabbitAutoConfiguration javadoc
ReactiveElasticsearchClientAutoConfiguration javadoc
834
Configuration Class Links
ReactiveElasticsearchRepositoriesAutoConfiguration javadoc
ReactiveMultipartAutoConfiguration javadoc
ReactiveOAuth2ClientAutoConfiguration javadoc
ReactiveOAuth2ResourceServerAutoConfiguration javadoc
ReactiveSecurityAutoConfiguration javadoc
ReactiveUserDetailsServiceAutoConfiguration javadoc
ReactiveWebServerFactoryAutoConfiguration javadoc
RedisAutoConfiguration javadoc
RedisReactiveAutoConfiguration javadoc
RedisRepositoriesAutoConfiguration javadoc
RepositoryRestMvcAutoConfiguration javadoc
RestTemplateAutoConfiguration javadoc
Saml2RelyingPartyAutoConfiguration javadoc
SecurityAutoConfiguration javadoc
SecurityFilterAutoConfiguration javadoc
SendGridAutoConfiguration javadoc
ServletWebServerFactoryAutoConfiguration javadoc
SessionAutoConfiguration javadoc
SpringApplicationAdminJmxAutoConfiguration javadoc
SpringDataWebAutoConfiguration javadoc
SqlInitializationAutoConfiguration javadoc
SslAutoConfiguration javadoc
TaskExecutionAutoConfiguration javadoc
TaskSchedulingAutoConfiguration javadoc
ThymeleafAutoConfiguration javadoc
TransactionAutoConfiguration javadoc
UserDetailsServiceAutoConfiguration javadoc
ValidationAutoConfiguration javadoc
WebClientAutoConfiguration javadoc
WebFluxAutoConfiguration javadoc
WebMvcAutoConfiguration javadoc
WebServiceTemplateAutoConfiguration javadoc
WebServicesAutoConfiguration javadoc
835
Configuration Class Links
WebSessionIdResolverAutoConfiguration javadoc
WebSocketMessagingAutoConfiguration javadoc
WebSocketReactiveAutoConfiguration javadoc
WebSocketServletAutoConfiguration javadoc
XADataSourceAutoConfiguration javadoc
.C.2. spring-boot-actuator-autoconfigure
836
Configuration Class Links
DiskSpaceHealthContributorAutoConfiguration javadoc
DynatraceMetricsExportAutoConfiguration javadoc
ElasticMetricsExportAutoConfiguration javadoc
ElasticsearchReactiveHealthContributorAutoConfiguration javadoc
ElasticsearchRestHealthContributorAutoConfiguration javadoc
EndpointAutoConfiguration javadoc
EnvironmentEndpointAutoConfiguration javadoc
FlywayEndpointAutoConfiguration javadoc
GangliaMetricsExportAutoConfiguration javadoc
GraphQlObservationAutoConfiguration javadoc
GraphiteMetricsExportAutoConfiguration javadoc
HazelcastHealthContributorAutoConfiguration javadoc
HealthContributorAutoConfiguration javadoc
HealthEndpointAutoConfiguration javadoc
HeapDumpWebEndpointAutoConfiguration javadoc
HibernateMetricsAutoConfiguration javadoc
HttpClientObservationsAutoConfiguration javadoc
HttpExchangesAutoConfiguration javadoc
HttpExchangesEndpointAutoConfiguration javadoc
HumioMetricsExportAutoConfiguration javadoc
InfluxDbHealthContributorAutoConfiguration javadoc
InfluxMetricsExportAutoConfiguration javadoc
InfoContributorAutoConfiguration javadoc
InfoEndpointAutoConfiguration javadoc
IntegrationGraphEndpointAutoConfiguration javadoc
JacksonEndpointAutoConfiguration javadoc
JerseyServerMetricsAutoConfiguration javadoc
JettyMetricsAutoConfiguration javadoc
JmsHealthContributorAutoConfiguration javadoc
JmxEndpointAutoConfiguration javadoc
JmxMetricsExportAutoConfiguration javadoc
JvmMetricsAutoConfiguration javadoc
KafkaMetricsAutoConfiguration javadoc
837
Configuration Class Links
KairosMetricsExportAutoConfiguration javadoc
LdapHealthContributorAutoConfiguration javadoc
LettuceMetricsAutoConfiguration javadoc
LiquibaseEndpointAutoConfiguration javadoc
Log4J2MetricsAutoConfiguration javadoc
LogFileWebEndpointAutoConfiguration javadoc
LogbackMetricsAutoConfiguration javadoc
LoggersEndpointAutoConfiguration javadoc
MailHealthContributorAutoConfiguration javadoc
ManagementContextAutoConfiguration javadoc
ManagementWebSecurityAutoConfiguration javadoc
MappingsEndpointAutoConfiguration javadoc
MetricsAutoConfiguration javadoc
MetricsEndpointAutoConfiguration javadoc
MicrometerTracingAutoConfiguration javadoc
MongoHealthContributorAutoConfiguration javadoc
MongoMetricsAutoConfiguration javadoc
MongoReactiveHealthContributorAutoConfiguration javadoc
Neo4jHealthContributorAutoConfiguration javadoc
NewRelicMetricsExportAutoConfiguration javadoc
ObservationAutoConfiguration javadoc
OpenTelemetryAutoConfiguration javadoc
OtlpAutoConfiguration javadoc
OtlpMetricsExportAutoConfiguration javadoc
PrometheusExemplarsAutoConfiguration javadoc
PrometheusMetricsExportAutoConfiguration javadoc
QuartzEndpointAutoConfiguration javadoc
RabbitHealthContributorAutoConfiguration javadoc
RabbitMetricsAutoConfiguration javadoc
ReactiveCloudFoundryActuatorAutoConfiguration javadoc
ReactiveManagementContextAutoConfiguration javadoc
ReactiveManagementWebSecurityAutoConfiguration javadoc
RedisHealthContributorAutoConfiguration javadoc
838
Configuration Class Links
RedisReactiveHealthContributorAutoConfiguration javadoc
RepositoryMetricsAutoConfiguration javadoc
ScheduledTasksEndpointAutoConfiguration javadoc
ServletManagementContextAutoConfiguration javadoc
SessionsEndpointAutoConfiguration javadoc
ShutdownEndpointAutoConfiguration javadoc
SignalFxMetricsExportAutoConfiguration javadoc
SimpleMetricsExportAutoConfiguration javadoc
StackdriverMetricsExportAutoConfiguration javadoc
StartupEndpointAutoConfiguration javadoc
StartupTimeMetricsListenerAutoConfiguration javadoc
StatsdMetricsExportAutoConfiguration javadoc
SystemMetricsAutoConfiguration javadoc
TaskExecutorMetricsAutoConfiguration javadoc
ThreadDumpEndpointAutoConfiguration javadoc
TomcatMetricsAutoConfiguration javadoc
WavefrontAutoConfiguration javadoc
WavefrontMetricsExportAutoConfiguration javadoc
WavefrontTracingAutoConfiguration javadoc
WebEndpointAutoConfiguration javadoc
WebFluxObservationAutoConfiguration javadoc
WebMvcObservationAutoConfiguration javadoc
ZipkinAutoConfiguration javadoc
The following table lists the various @…Test annotations that can be used to test slices of your
application and the auto-configuration that they import by default:
839
Test slice Imported auto-configuration
@DataCassandraTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.cassand
ra.CassandraAutoConfiguration
org.springframework.boot.autoconfigure.data.ca
ssandra.CassandraDataAutoConfiguration
org.springframework.boot.autoconfigure.data.ca
ssandra.CassandraReactiveDataAutoConfiguration
org.springframework.boot.autoconfigure.data.ca
ssandra.CassandraReactiveRepositoriesAutoConfi
guration
org.springframework.boot.autoconfigure.data.ca
ssandra.CassandraRepositoriesAutoConfiguration
@DataCouchbaseTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.couchba
se.CouchbaseAutoConfiguration
org.springframework.boot.autoconfigure.data.co
uchbase.CouchbaseDataAutoConfiguration
org.springframework.boot.autoconfigure.data.co
uchbase.CouchbaseReactiveDataAutoConfiguration
org.springframework.boot.autoconfigure.data.co
uchbase.CouchbaseReactiveRepositoriesAutoConfi
guration
org.springframework.boot.autoconfigure.data.co
uchbase.CouchbaseRepositoriesAutoConfiguration
@DataElasticsearchTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.el
asticsearch.ElasticsearchDataAutoConfiguration
org.springframework.boot.autoconfigure.data.el
asticsearch.ElasticsearchRepositoriesAutoConfi
guration
org.springframework.boot.autoconfigure.data.el
asticsearch.ReactiveElasticsearchRepositoriesA
utoConfiguration
org.springframework.boot.autoconfigure.elastic
search.ElasticsearchClientAutoConfiguration
org.springframework.boot.autoconfigure.elastic
search.ElasticsearchRestClientAutoConfiguratio
n
org.springframework.boot.autoconfigure.elastic
search.ReactiveElasticsearchClientAutoConfigur
ation
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
840
Test slice Imported auto-configuration
@DataJdbcTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.jd
bc.JdbcRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.flyway.
FlywayAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceTransactionManagerAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Jd
bcTemplateAutoConfiguration
org.springframework.boot.autoconfigure.liquiba
se.LiquibaseAutoConfiguration
org.springframework.boot.autoconfigure.sql.ini
t.SqlInitializationAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
org.springframework.boot.test.autoconfigure.jd
bc.TestDatabaseAutoConfiguration
@DataJpaTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.jp
a.JpaRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.flyway.
FlywayAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceTransactionManagerAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Jd
bcTemplateAutoConfiguration
org.springframework.boot.autoconfigure.liquiba
se.LiquibaseAutoConfiguration
org.springframework.boot.autoconfigure.orm.jpa
.HibernateJpaAutoConfiguration
org.springframework.boot.autoconfigure.sql.ini
t.SqlInitializationAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
org.springframework.boot.test.autoconfigure.jd
bc.TestDatabaseAutoConfiguration
org.springframework.boot.test.autoconfigure.or
m.jpa.TestEntityManagerAutoConfiguration
841
Test slice Imported auto-configuration
@DataLdapTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.ld
ap.LdapRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.ldap.Ld
apAutoConfiguration
org.springframework.boot.autoconfigure.ldap.em
bedded.EmbeddedLdapAutoConfiguration
@DataMongoTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.mo
ngo.MongoDataAutoConfiguration
org.springframework.boot.autoconfigure.data.mo
ngo.MongoReactiveDataAutoConfiguration
org.springframework.boot.autoconfigure.data.mo
ngo.MongoReactiveRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.data.mo
ngo.MongoRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.mongo.M
ongoAutoConfiguration
org.springframework.boot.autoconfigure.mongo.M
ongoReactiveAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
@DataNeo4jTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.ne
o4j.Neo4jDataAutoConfiguration
org.springframework.boot.autoconfigure.data.ne
o4j.Neo4jReactiveDataAutoConfiguration
org.springframework.boot.autoconfigure.data.ne
o4j.Neo4jReactiveRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.data.ne
o4j.Neo4jRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.neo4j.N
eo4jAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
842
Test slice Imported auto-configuration
@DataR2dbcTest org.springframework.boot.autoconfigure.data.r2
dbc.R2dbcDataAutoConfiguration
org.springframework.boot.autoconfigure.data.r2
dbc.R2dbcRepositoriesAutoConfiguration
org.springframework.boot.autoconfigure.flyway.
FlywayAutoConfiguration
org.springframework.boot.autoconfigure.liquiba
se.LiquibaseAutoConfiguration
org.springframework.boot.autoconfigure.r2dbc.R
2dbcAutoConfiguration
org.springframework.boot.autoconfigure.r2dbc.R
2dbcTransactionManagerAutoConfiguration
org.springframework.boot.autoconfigure.sql.ini
t.SqlInitializationAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
@DataRedisTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.data.re
dis.RedisAutoConfiguration
org.springframework.boot.autoconfigure.data.re
dis.RedisReactiveAutoConfiguration
org.springframework.boot.autoconfigure.data.re
dis.RedisRepositoriesAutoConfiguration
@GraphQlTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.graphql
.GraphQlAutoConfiguration
org.springframework.boot.autoconfigure.gson.Gs
onAutoConfiguration
org.springframework.boot.autoconfigure.http.co
dec.CodecsAutoConfiguration
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
org.springframework.boot.autoconfigure.validat
ion.ValidationAutoConfiguration
org.springframework.boot.test.autoconfigure.gr
aphql.tester.GraphQlTesterAutoConfiguration
843
Test slice Imported auto-configuration
@JdbcTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.flyway.
FlywayAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceTransactionManagerAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Jd
bcTemplateAutoConfiguration
org.springframework.boot.autoconfigure.liquiba
se.LiquibaseAutoConfiguration
org.springframework.boot.autoconfigure.sql.ini
t.SqlInitializationAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
org.springframework.boot.test.autoconfigure.jd
bc.TestDatabaseAutoConfiguration
@JooqTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.flyway.
FlywayAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceAutoConfiguration
org.springframework.boot.autoconfigure.jdbc.Da
taSourceTransactionManagerAutoConfiguration
org.springframework.boot.autoconfigure.jooq.Jo
oqAutoConfiguration
org.springframework.boot.autoconfigure.liquiba
se.LiquibaseAutoConfiguration
org.springframework.boot.autoconfigure.sql.ini
t.SqlInitializationAutoConfiguration
org.springframework.boot.autoconfigure.transac
tion.TransactionAutoConfiguration
@JsonTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.gson.Gs
onAutoConfiguration
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
org.springframework.boot.test.autoconfigure.js
on.JsonTestersAutoConfiguration
844
Test slice Imported auto-configuration
@RestClientTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.gson.Gs
onAutoConfiguration
org.springframework.boot.autoconfigure.http.Ht
tpMessageConvertersAutoConfiguration
org.springframework.boot.autoconfigure.http.co
dec.CodecsAutoConfiguration
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
org.springframework.boot.autoconfigure.web.cli
ent.RestTemplateAutoConfiguration
org.springframework.boot.autoconfigure.web.rea
ctive.function.client.WebClientAutoConfigurati
on
org.springframework.boot.test.autoconfigure.we
b.client.MockRestServiceServerAutoConfiguratio
n
org.springframework.boot.test.autoconfigure.we
b.client.WebClientRestTemplateAutoConfiguratio
n
845
Test slice Imported auto-configuration
@WebFluxTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.context
.MessageSourceAutoConfiguration
org.springframework.boot.autoconfigure.freemar
ker.FreeMarkerAutoConfiguration
org.springframework.boot.autoconfigure.gson.Gs
onAutoConfiguration
org.springframework.boot.autoconfigure.http.co
dec.CodecsAutoConfiguration
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
org.springframework.boot.autoconfigure.mustach
e.MustacheAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.oauth2.client.reactive.ReactiveOAuth2ClientA
utoConfiguration
org.springframework.boot.autoconfigure.securit
y.oauth2.resource.reactive.ReactiveOAuth2Resou
rceServerAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.reactive.ReactiveSecurityAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.reactive.ReactiveUserDetailsServiceAutoConfi
guration
org.springframework.boot.autoconfigure.thymele
af.ThymeleafAutoConfiguration
org.springframework.boot.autoconfigure.validat
ion.ValidationAutoConfiguration
org.springframework.boot.autoconfigure.web.rea
ctive.WebFluxAutoConfiguration
org.springframework.boot.autoconfigure.web.rea
ctive.error.ErrorWebFluxAutoConfiguration
org.springframework.boot.test.autoconfigure.we
b.reactive.WebTestClientAutoConfiguration
846
Test slice Imported auto-configuration
@WebMvcTest org.springframework.boot.autoconfigure.cache.C
acheAutoConfiguration
org.springframework.boot.autoconfigure.context
.MessageSourceAutoConfiguration
org.springframework.boot.autoconfigure.data.we
b.SpringDataWebAutoConfiguration
org.springframework.boot.autoconfigure.freemar
ker.FreeMarkerAutoConfiguration
org.springframework.boot.autoconfigure.groovy.
template.GroovyTemplateAutoConfiguration
org.springframework.boot.autoconfigure.gson.Gs
onAutoConfiguration
org.springframework.boot.autoconfigure.hateoas
.HypermediaAutoConfiguration
org.springframework.boot.autoconfigure.http.Ht
tpMessageConvertersAutoConfiguration
org.springframework.boot.autoconfigure.jackson
.JacksonAutoConfiguration
org.springframework.boot.autoconfigure.jsonb.J
sonbAutoConfiguration
org.springframework.boot.autoconfigure.mustach
e.MustacheAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.oauth2.client.servlet.OAuth2ClientAutoConfig
uration
org.springframework.boot.autoconfigure.securit
y.oauth2.resource.servlet.OAuth2ResourceServer
AutoConfiguration
org.springframework.boot.autoconfigure.securit
y.servlet.SecurityAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.servlet.SecurityFilterAutoConfiguration
org.springframework.boot.autoconfigure.securit
y.servlet.UserDetailsServiceAutoConfiguration
org.springframework.boot.autoconfigure.task.Ta
skExecutionAutoConfiguration
org.springframework.boot.autoconfigure.thymele
af.ThymeleafAutoConfiguration
org.springframework.boot.autoconfigure.validat
ion.ValidationAutoConfiguration
org.springframework.boot.autoconfigure.web.ser
vlet.HttpEncodingAutoConfiguration
org.springframework.boot.autoconfigure.web.ser
vlet.WebMvcAutoConfiguration
org.springframework.boot.autoconfigure.web.ser
vlet.error.ErrorMvcAutoConfiguration
org.springframework.boot.test.autoconfigure.we
b.reactive.WebTestClientAutoConfiguration
org.springframework.boot.test.autoconfigure.we
b.servlet.MockMvcAutoConfiguration
org.springframework.boot.test.autoconfigure.we
b.servlet.MockMvcSecurityConfiguration
org.springframework.boot.test.autoconfigure.we
b.servlet.MockMvcWebClientAutoConfiguration 847
Test slice Imported auto-configuration
@WebServiceClientTest org.springframework.boot.test.autoconfigure.we
org.springframework.boot.autoconfigure.cache.C
b.servlet.MockMvcWebDriverAutoConfiguration
acheAutoConfiguration
org.springframework.boot.autoconfigure.webserv
ices.client.WebServiceTemplateAutoConfiguratio
n
org.springframework.boot.test.autoconfigure.we
bservices.client.MockWebServiceServerAutoConfi
guration
org.springframework.boot.test.autoconfigure.we
bservices.client.WebServiceClientTemplateAutoC
onfiguration
@WebServiceServerTest org.springframework.boot.autoconfigure.webserv
ices.WebServicesAutoConfiguration
org.springframework.boot.test.autoconfigure.we
bservices.server.MockWebServiceClientAutoConfi
guration
If you need to create executable jars from a different build system or if you are just curious about
the underlying technology, this appendix provides some background.
Java does not provide any standard way to load nested jar files (that is, jar files that are themselves
contained within a jar). This can be problematic if you need to distribute a self-contained
application that can be run from the command line without unpacking.
To solve this problem, many developers use “shaded” jars. A shaded jar packages all classes, from
all jars, into a single “uber jar”. The problem with shaded jars is that it becomes hard to see which
libraries are actually in your application. It can also be problematic if the same filename is used
(but with different content) in multiple jars. Spring Boot takes a different approach and lets you
actually nest jars directly.
Spring Boot Loader-compatible jar files should be structured in the following way:
848
example.jar
|
+-META-INF
| +-MANIFEST.MF
+-org
| +-springframework
| +-boot
| +-loader
| +-<spring boot loader classes>
+-BOOT-INF
+-classes
| +-mycompany
| +-project
| +-YourClasses.class
+-lib
+-dependency1.jar
+-dependency2.jar
Spring Boot Loader-compatible war files should be structured in the following way:
example.war
|
+-META-INF
| +-MANIFEST.MF
+-org
| +-springframework
| +-boot
| +-loader
| +-<spring boot loader classes>
+-WEB-INF
+-classes
| +-com
| +-mycompany
| +-project
| +-YourClasses.class
+-lib
| +-dependency1.jar
| +-dependency2.jar
+-lib-provided
+-servlet-api.jar
+-dependency3.jar
Dependencies should be placed in a nested WEB-INF/lib directory. Any dependencies that are
849
required when running embedded but are not required when deploying to a traditional web
container should be placed in WEB-INF/lib-provided.
Index Files
Spring Boot Loader-compatible jar and war archives can include additional index files under the
BOOT-INF/ directory. A classpath.idx file can be provided for both jars and wars, and it provides the
ordering that jars should be added to the classpath. The layers.idx file can be used only for jars,
and it allows a jar to be split into logical layers for Docker/OCI image creation.
Index files follow a YAML compatible syntax so that they can be easily parsed by third-party tools.
These files, however, are not parsed internally as YAML and they must be written in exactly the
formats described below in order to be used.
Classpath Index
The classpath index file can be provided in BOOT-INF/classpath.idx. It provides a list of jar names
(including the directory) in the order that they should be added to the classpath. Each line must
start with dash space ("-·") and names must be in double quotes.
example.jar
|
+-META-INF
| +-...
+-BOOT-INF
+-classes
| +...
+-lib
+-dependency1.jar
+-dependency2.jar
- "BOOT-INF/lib/dependency2.jar"
- "BOOT-INF/lib/dependency1.jar"
Layer Index
The layers index file can be provided in BOOT-INF/layers.idx. It provides a list of layers and the
parts of the jar that should be contained within them. Layers are written in the order that they
should be added to the Docker/OCI image. Layers names are written as quoted strings prefixed with
dash space ("-·") and with a colon (":") suffix. Layer content is either a file or directory name
written as a quoted string prefixed by space space dash space ("··-·"). A directory name ends with
/, a file name does not. When a directory name is used it means that all files inside that directory
are in the same layer.
850
A typical example of a layers index would be:
- "dependencies":
- "BOOT-INF/lib/dependency1.jar"
- "BOOT-INF/lib/dependency2.jar"
- "application":
- "BOOT-INF/classes/"
- "META-INF/"
myapp.jar
+-------------------+-------------------------+
| /BOOT-INF/classes | /BOOT-INF/lib/mylib.jar |
|+-----------------+||+-----------+----------+|
|| A.class ||| B.class | C.class ||
|+-----------------+||+-----------+----------+|
+-------------------+-------------------------+
^ ^ ^
0063 3452 3980
The preceding example shows how A.class can be found in /BOOT-INF/classes in myapp.jar at
position 0063. B.class from the nested jar can actually be found in myapp.jar at position 3452, and
C.class is at position 3980.
Armed with this information, we can load specific nested entries by seeking to the appropriate part
of the outer jar. We do not need to unpack the archive, and we do not need to read all entry data
into memory.
Spring Boot Loader strives to remain compatible with existing code and libraries.
org.springframework.boot.loader.jar.JarFile extends from java.util.jar.JarFile and should work
as a drop-in replacement. The getURL() method returns a URL that opens a connection compatible
with java.net.JarURLConnection and can be used with Java’s URLClassLoader.
There are three launcher subclasses (JarLauncher, WarLauncher, and PropertiesLauncher). Their
851
purpose is to load resources (.class files and so on) from nested jar files or war files in directories
(as opposed to those explicitly on the classpath). In the case of JarLauncher and WarLauncher, the
nested paths are fixed. JarLauncher looks in BOOT-INF/lib/, and WarLauncher looks in WEB-INF/lib/
and WEB-INF/lib-provided/. You can add extra jars in those locations if you want more. The
PropertiesLauncher looks in BOOT-INF/lib/ in your application archive by default. You can add
additional locations by setting an environment variable called LOADER_PATH or loader.path in
loader.properties (which is a comma-separated list of directories, archives, or directories within
archives).
Launcher Manifest
You need to specify an appropriate Launcher as the Main-Class attribute of META-INF/MANIFEST.MF. The
actual class that you want to launch (that is, the class that contains a main method) should be
specified in the Start-Class attribute.
The following example shows a typical MANIFEST.MF for an executable jar file:
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.mycompany.project.MyApplication
Main-Class: org.springframework.boot.loader.WarLauncher
Start-Class: com.mycompany.project.MyApplication
You need not specify Class-Path entries in your manifest file. The classpath is
NOTE
deduced from the nested jars.
PropertiesLauncher has a few special features that can be enabled with external properties (System
properties, environment variables, manifest entries, or loader.properties). The following table
describes these properties:
Key Purpose
loader.path Comma-separated Classpath, such as
lib,${HOME}/app/lib. Earlier entries take
precedence, like a regular -classpath on the
javac command line.
852
Key Purpose
loader.home Used to resolve relative paths in loader.path. For
example, given loader.path=lib, then
${loader.home}/lib is a classpath location (along
with all jar files in that directory). This property
is also used to locate a loader.properties file, as
in the following example /opt/app It defaults to
${user.dir}.
loader.args Default arguments for the main method (space
separated).
loader.main Name of main class to launch (for example,
com.app.Application).
loader.config.name Name of properties file (for example, launcher).
It defaults to loader.
loader.config.location Path to properties file (for example,
classpath:loader.properties). It defaults to
loader.properties.
loader.system Boolean flag to indicate that all properties
should be added to System properties. It defaults
to false.
When specified as environment variables or manifest entries, the following names should be used:
Build plugins automatically move the Main-Class attribute to Start-Class when the fat
TIP jar is built. If you use that, specify the name of the class to launch by using the Main-
Class attribute and leaving out Start-Class.
• loader.properties is searched for in loader.home, then in the root of the classpath, and then in
classpath:/BOOT-INF/classes. The first location where a file with that name exists is used.
• loader.home is the directory location of an additional properties file (overriding the default) only
when loader.config.location is not specified.
• loader.path can contain directories (which are scanned recursively for jar and zip files), archive
paths, a directory within an archive that is scanned for jar files (for example,
853
dependencies.jar!/lib), or wildcard patterns (for the default JVM behavior). Archive paths can
be relative to loader.home or anywhere in the file system with a jar:file: prefix.
• loader.path (if empty) defaults to BOOT-INF/lib (meaning a local directory or a nested one if
running from an archive). Because of this, PropertiesLauncher behaves the same as JarLauncher
when no additional configuration is provided.
• loader.path can not be used to configure the location of loader.properties (the classpath used to
search for the latter is the JVM classpath when PropertiesLauncher is launched).
• Placeholder replacement is done from System and environment variables plus the properties
file itself on all values before use.
• The search order for properties (where it makes sense to look in more than one place) is
environment variables, system properties, loader.properties, the exploded archive manifest,
and the archive manifest.
You need to consider the following restrictions when working with a Spring Boot Loader packaged
application:
• Zip entry compression: The ZipEntry for a nested jar must be saved by using the ZipEntry.STORED
method. This is required so that we can seek directly to individual content within the nested jar.
The content of the nested jar file itself can still be compressed, as can any other entries in the
outer jar.
If the preceding restrictions mean that you cannot use Spring Boot Loader, consider the following
alternatives:
• JarClassLoader
• OneJar
The following table provides details of all of the dependency versions that are provided by Spring
Boot in its CLI (Command Line Interface), Maven dependency management, and Gradle plugin.
854
When you declare a dependency on one of these artifacts without declaring a version, the version
listed in the table is used.
855
Group ID Artifact ID Version
com.fasterxml.jackson.datatype jackson-datatype-eclipse- 2.15.0
collections
com.fasterxml.jackson.datatype jackson-datatype-guava 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-hibernate4 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-hibernate5 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-hibernate5- 2.15.0
jakarta
com.fasterxml.jackson.datatype jackson-datatype-hibernate6 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-hppc 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-jakarta-jsonp 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-jaxrs 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-jdk8 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-joda 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-joda-money 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-json-org 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-jsr310 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-jsr353 2.15.0
com.fasterxml.jackson.datatype jackson-datatype-pcollections 2.15.0
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-base 2.15.0
rs
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-cbor- 2.15.0
rs provider
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-json- 2.15.0
rs provider
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-smile- 2.15.0
rs provider
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-xml- 2.15.0
rs provider
com.fasterxml.jackson.jakarta. jackson-jakarta-rs-yaml- 2.15.0
rs provider
com.fasterxml.jackson.jaxrs jackson-jaxrs-base 2.15.0
com.fasterxml.jackson.jaxrs jackson-jaxrs-cbor-provider 2.15.0
com.fasterxml.jackson.jaxrs jackson-jaxrs-json-provider 2.15.0
com.fasterxml.jackson.jaxrs jackson-jaxrs-smile-provider 2.15.0
com.fasterxml.jackson.jaxrs jackson-jaxrs-xml-provider 2.15.0
com.fasterxml.jackson.jaxrs jackson-jaxrs-yaml-provider 2.15.0
com.fasterxml.jackson.jr jackson-jr-all 2.15.0
com.fasterxml.jackson.jr jackson-jr-annotation-support 2.15.0
com.fasterxml.jackson.jr jackson-jr-objects 2.15.0
com.fasterxml.jackson.jr jackson-jr-retrofit2 2.15.0
com.fasterxml.jackson.jr jackson-jr-stree 2.15.0
856
Group ID Artifact ID Version
com.fasterxml.jackson.module jackson-module-afterburner 2.15.0
com.fasterxml.jackson.module jackson-module-blackbird 2.15.0
com.fasterxml.jackson.module jackson-module-guice 2.15.0
com.fasterxml.jackson.module jackson-module-jakarta- 2.15.0
xmlbind-annotations
com.fasterxml.jackson.module jackson-module-jaxb- 2.15.0
annotations
com.fasterxml.jackson.module jackson-module-jsonSchema 2.15.0
com.fasterxml.jackson.module jackson-module-jsonSchema- 2.15.0
jakarta
com.fasterxml.jackson.module jackson-module-kotlin 2.15.0
com.fasterxml.jackson.module jackson-module-mrbean 2.15.0
com.fasterxml.jackson.module jackson-module-no-ctor-deser 2.15.0
com.fasterxml.jackson.module jackson-module-osgi 2.15.0
com.fasterxml.jackson.module jackson-module-parameter-names 2.15.0
com.fasterxml.jackson.module jackson-module-paranamer 2.15.0
com.fasterxml.jackson.module jackson-module-scala_2.11 2.15.0
com.fasterxml.jackson.module jackson-module-scala_2.12 2.15.0
com.fasterxml.jackson.module jackson-module-scala_2.13 2.15.0
com.fasterxml.jackson.module jackson-module-scala_3 2.15.0
com.github.ben-manes.caffeine caffeine 3.1.6
com.github.ben-manes.caffeine guava 3.1.6
com.github.ben-manes.caffeine jcache 3.1.6
com.github.ben-manes.caffeine simulator 3.1.6
com.github.mxab.thymeleaf.extr thymeleaf-extras-data- 2.0.1
as attribute
com.google.code.gson gson 2.10.1
com.graphql-java graphql-java 20.2
com.h2database h2 2.1.214
com.hazelcast hazelcast 5.2.3
com.hazelcast hazelcast-spring 5.2.3
com.ibm.db2 jcc 11.5.8.0
com.jayway.jsonpath json-path 2.8.0
com.jayway.jsonpath json-path-assert 2.8.0
com.microsoft.sqlserver mssql-jdbc 11.2.3.jre17
com.mysql mysql-connector-j 8.0.33
com.oracle.database.ha ons 21.9.0.0
com.oracle.database.ha simplefan 21.9.0.0
com.oracle.database.jdbc ojdbc11 21.9.0.0
com.oracle.database.jdbc ojdbc11-production 21.9.0.0
857
Group ID Artifact ID Version
com.oracle.database.jdbc ojdbc8 21.9.0.0
com.oracle.database.jdbc ojdbc8-production 21.9.0.0
com.oracle.database.jdbc rsi 21.9.0.0
com.oracle.database.jdbc ucp 21.9.0.0
com.oracle.database.jdbc ucp11 21.9.0.0
com.oracle.database.jdbc.debug ojdbc11-debug 21.9.0.0
com.oracle.database.jdbc.debug ojdbc11-observability-debug 21.9.0.0
com.oracle.database.jdbc.debug ojdbc11_g 21.9.0.0
com.oracle.database.jdbc.debug ojdbc11dms_g 21.9.0.0
com.oracle.database.jdbc.debug ojdbc8-debug 21.9.0.0
com.oracle.database.jdbc.debug ojdbc8-observability-debug 21.9.0.0
com.oracle.database.jdbc.debug ojdbc8_g 21.9.0.0
com.oracle.database.jdbc.debug ojdbc8dms_g 21.9.0.0
com.oracle.database.nls orai18n 21.9.0.0
com.oracle.database.observabil dms 21.9.0.0
ity
com.oracle.database.observabil ojdbc11-observability 21.9.0.0
ity
com.oracle.database.observabil ojdbc11dms 21.9.0.0
ity
com.oracle.database.observabil ojdbc8-observability 21.9.0.0
ity
com.oracle.database.observabil ojdbc8dms 21.9.0.0
ity
com.oracle.database.r2dbc oracle-r2dbc 1.1.1
com.oracle.database.security oraclepki 21.9.0.0
com.oracle.database.security osdt_cert 21.9.0.0
com.oracle.database.security osdt_core 21.9.0.0
com.oracle.database.xml xdb 21.9.0.0
com.oracle.database.xml xmlparserv2 21.9.0.0
com.querydsl querydsl-apt 5.0.0
com.querydsl querydsl-codegen 5.0.0
com.querydsl querydsl-codegen-utils 5.0.0
com.querydsl querydsl-collections 5.0.0
com.querydsl querydsl-core 5.0.0
com.querydsl querydsl-guava 5.0.0
com.querydsl querydsl-hibernate-search 5.0.0
com.querydsl querydsl-jdo 5.0.0
com.querydsl querydsl-jpa 5.0.0
com.querydsl querydsl-jpa-codegen 5.0.0
858
Group ID Artifact ID Version
com.querydsl querydsl-kotlin 5.0.0
com.querydsl querydsl-kotlin-codegen 5.0.0
com.querydsl querydsl-lucene3 5.0.0
com.querydsl querydsl-lucene4 5.0.0
com.querydsl querydsl-lucene5 5.0.0
com.querydsl querydsl-mongodb 5.0.0
com.querydsl querydsl-scala 5.0.0
com.querydsl querydsl-spatial 5.0.0
com.querydsl querydsl-sql 5.0.0
com.querydsl querydsl-sql-codegen 5.0.0
com.querydsl querydsl-sql-spatial 5.0.0
com.querydsl querydsl-sql-spring 5.0.0
com.rabbitmq amqp-client 5.17.0
com.rabbitmq stream-client 0.9.0
com.samskivert jmustache 1.15
com.sendgrid sendgrid-java 4.9.3
com.squareup.okhttp3 logging-interceptor 4.10.0
com.squareup.okhttp3 mockwebserver 4.10.0
com.squareup.okhttp3 okcurl 4.10.0
com.squareup.okhttp3 okhttp 4.10.0
com.squareup.okhttp3 okhttp-brotli 4.10.0
com.squareup.okhttp3 okhttp-dnsoverhttps 4.10.0
com.squareup.okhttp3 okhttp-sse 4.10.0
com.squareup.okhttp3 okhttp-tls 4.10.0
com.squareup.okhttp3 okhttp-urlconnection 4.10.0
com.sun.istack istack-commons-runtime 4.1.1
com.sun.xml.bind jaxb-core 4.0.2
com.sun.xml.bind jaxb-impl 4.0.2
com.sun.xml.bind jaxb-jxc 4.0.2
com.sun.xml.bind jaxb-osgi 4.0.2
com.sun.xml.bind jaxb-xjc 4.0.2
com.sun.xml.fastinfoset FastInfoset 2.1.0
com.sun.xml.messaging.saaj saaj-impl 3.0.1
com.unboundid unboundid-ldapsdk 6.0.8
com.zaxxer HikariCP 5.0.1
commons-codec commons-codec 1.15
commons-pool commons-pool 1.6
io.asyncer r2dbc-mysql 1.0.1
859
Group ID Artifact ID Version
io.dropwizard.metrics metrics-annotation 4.2.18
io.dropwizard.metrics metrics-caffeine 4.2.18
io.dropwizard.metrics metrics-caffeine3 4.2.18
io.dropwizard.metrics metrics-collectd 4.2.18
io.dropwizard.metrics metrics-core 4.2.18
io.dropwizard.metrics metrics-ehcache 4.2.18
io.dropwizard.metrics metrics-graphite 4.2.18
io.dropwizard.metrics metrics-healthchecks 4.2.18
io.dropwizard.metrics metrics-httpasyncclient 4.2.18
io.dropwizard.metrics metrics-httpclient 4.2.18
io.dropwizard.metrics metrics-httpclient5 4.2.18
io.dropwizard.metrics metrics-jakarta-servlet 4.2.18
io.dropwizard.metrics metrics-jakarta-servlets 4.2.18
io.dropwizard.metrics metrics-jcache 4.2.18
io.dropwizard.metrics metrics-jdbi 4.2.18
io.dropwizard.metrics metrics-jdbi3 4.2.18
io.dropwizard.metrics metrics-jersey2 4.2.18
io.dropwizard.metrics metrics-jersey3 4.2.18
io.dropwizard.metrics metrics-jersey31 4.2.18
io.dropwizard.metrics metrics-jetty10 4.2.18
io.dropwizard.metrics metrics-jetty11 4.2.18
io.dropwizard.metrics metrics-jetty9 4.2.18
io.dropwizard.metrics metrics-jmx 4.2.18
io.dropwizard.metrics metrics-json 4.2.18
io.dropwizard.metrics metrics-jvm 4.2.18
io.dropwizard.metrics metrics-log4j2 4.2.18
io.dropwizard.metrics metrics-logback 4.2.18
io.dropwizard.metrics metrics-logback13 4.2.18
io.dropwizard.metrics metrics-logback14 4.2.18
io.dropwizard.metrics metrics-servlet 4.2.18
io.dropwizard.metrics metrics-servlets 4.2.18
io.lettuce lettuce-core 6.2.4.RELEASE
io.micrometer micrometer-commons 1.11.0-RC1
io.micrometer micrometer-core 1.11.0-RC1
io.micrometer micrometer-jetty11 1.11.0-RC1
io.micrometer micrometer-observation 1.11.0-RC1
io.micrometer micrometer-observation-test 1.11.0-RC1
io.micrometer micrometer-osgi-test 1.11.0-RC1
860
Group ID Artifact ID Version
io.micrometer micrometer-registry-appoptics 1.11.0-RC1
io.micrometer micrometer-registry-atlas 1.11.0-RC1
io.micrometer micrometer-registry-azure- 1.11.0-RC1
monitor
io.micrometer micrometer-registry-cloudwatch 1.11.0-RC1
io.micrometer micrometer-registry- 1.11.0-RC1
cloudwatch2
io.micrometer micrometer-registry-datadog 1.11.0-RC1
io.micrometer micrometer-registry-dynatrace 1.11.0-RC1
io.micrometer micrometer-registry-elastic 1.11.0-RC1
io.micrometer micrometer-registry-ganglia 1.11.0-RC1
io.micrometer micrometer-registry-graphite 1.11.0-RC1
io.micrometer micrometer-registry-health 1.11.0-RC1
io.micrometer micrometer-registry-humio 1.11.0-RC1
io.micrometer micrometer-registry-influx 1.11.0-RC1
io.micrometer micrometer-registry-jmx 1.11.0-RC1
io.micrometer micrometer-registry-kairos 1.11.0-RC1
io.micrometer micrometer-registry-new-relic 1.11.0-RC1
io.micrometer micrometer-registry-opentsdb 1.11.0-RC1
io.micrometer micrometer-registry-otlp 1.11.0-RC1
io.micrometer micrometer-registry-prometheus 1.11.0-RC1
io.micrometer micrometer-registry-signalfx 1.11.0-RC1
io.micrometer micrometer-registry- 1.11.0-RC1
stackdriver
io.micrometer micrometer-registry-statsd 1.11.0-RC1
io.micrometer micrometer-registry-wavefront 1.11.0-RC1
io.micrometer micrometer-test 1.11.0-RC1
io.micrometer micrometer-tracing 1.1.0-RC1
io.micrometer micrometer-tracing-bridge- 1.1.0-RC1
brave
io.micrometer micrometer-tracing-bridge-otel 1.1.0-RC1
io.micrometer micrometer-tracing- 1.1.0-RC1
integration-test
io.micrometer micrometer-tracing-reporter- 1.1.0-RC1
wavefront
io.micrometer micrometer-tracing-test 1.1.0-RC1
io.netty netty-all 4.1.92.Final
io.netty netty-buffer 4.1.92.Final
io.netty netty-codec 4.1.92.Final
io.netty netty-codec-dns 4.1.92.Final
861
Group ID Artifact ID Version
io.netty netty-codec-haproxy 4.1.92.Final
io.netty netty-codec-http 4.1.92.Final
io.netty netty-codec-http2 4.1.92.Final
io.netty netty-codec-memcache 4.1.92.Final
io.netty netty-codec-mqtt 4.1.92.Final
io.netty netty-codec-redis 4.1.92.Final
io.netty netty-codec-smtp 4.1.92.Final
io.netty netty-codec-socks 4.1.92.Final
io.netty netty-codec-stomp 4.1.92.Final
io.netty netty-codec-xml 4.1.92.Final
io.netty netty-common 4.1.92.Final
io.netty netty-dev-tools 4.1.92.Final
io.netty netty-example 4.1.92.Final
io.netty netty-handler 4.1.92.Final
io.netty netty-handler-proxy 4.1.92.Final
io.netty netty-handler-ssl-ocsp 4.1.92.Final
io.netty netty-resolver 4.1.92.Final
io.netty netty-resolver-dns 4.1.92.Final
io.netty netty-resolver-dns-classes- 4.1.92.Final
macos
io.netty netty-resolver-dns-native- 4.1.92.Final
macos
io.netty netty-tcnative 2.0.60.Final
io.netty netty-tcnative-boringssl- 2.0.60.Final
static
io.netty netty-tcnative-classes 2.0.60.Final
io.netty netty-transport 4.1.92.Final
io.netty netty-transport-classes-epoll 4.1.92.Final
io.netty netty-transport-classes-kqueue 4.1.92.Final
io.netty netty-transport-native-epoll 4.1.92.Final
io.netty netty-transport-native-kqueue 4.1.92.Final
io.netty netty-transport-native-unix- 4.1.92.Final
common
io.netty netty-transport-rxtx 4.1.92.Final
io.netty netty-transport-sctp 4.1.92.Final
io.netty netty-transport-udt 4.1.92.Final
io.opentelemetry opentelemetry-api 1.25.0
io.opentelemetry opentelemetry-context 1.25.0
io.opentelemetry opentelemetry-exporter-common 1.25.0
io.opentelemetry opentelemetry-exporter-jaeger 1.25.0
862
Group ID Artifact ID Version
io.opentelemetry opentelemetry-exporter-jaeger- 1.17.0
proto
io.opentelemetry opentelemetry-exporter-jaeger- 1.25.0
thrift
io.opentelemetry opentelemetry-exporter-logging 1.25.0
io.opentelemetry opentelemetry-exporter- 1.25.0
logging-otlp
io.opentelemetry opentelemetry-exporter-otlp 1.25.0
io.opentelemetry opentelemetry-exporter-otlp- 1.25.0
common
io.opentelemetry opentelemetry-exporter-zipkin 1.25.0
io.opentelemetry opentelemetry-extension- 1.18.0
annotations
io.opentelemetry opentelemetry-extension-aws 1.20.1
io.opentelemetry opentelemetry-extension-kotlin 1.25.0
io.opentelemetry opentelemetry-extension-trace- 1.25.0
propagators
io.opentelemetry opentelemetry-sdk 1.25.0
io.opentelemetry opentelemetry-sdk-common 1.25.0
io.opentelemetry opentelemetry-sdk-extension- 1.25.0
autoconfigure-spi
io.opentelemetry opentelemetry-sdk-extension- 1.19.0
aws
io.opentelemetry opentelemetry-sdk-extension- 1.25.0
jaeger-remote-sampler
io.opentelemetry opentelemetry-sdk-extension- 1.19.0
resources
io.opentelemetry opentelemetry-sdk-metrics 1.25.0
io.opentelemetry opentelemetry-sdk-testing 1.25.0
io.opentelemetry opentelemetry-sdk-trace 1.25.0
io.projectreactor reactor-core 3.5.5
io.projectreactor reactor-core-micrometer 1.0.5
io.projectreactor reactor-test 3.5.5
io.projectreactor reactor-tools 3.5.5
io.projectreactor.addons reactor-adapter 3.5.1
io.projectreactor.addons reactor-extra 3.5.1
io.projectreactor.addons reactor-pool 1.0.0
io.projectreactor.addons reactor-pool-micrometer 0.1.0
io.projectreactor.kafka reactor-kafka 1.3.17
io.projectreactor.kotlin reactor-kotlin-extensions 1.2.2
io.projectreactor.netty reactor-netty 1.1.6
io.projectreactor.netty reactor-netty-core 1.1.6
863
Group ID Artifact ID Version
io.projectreactor.netty reactor-netty-http 1.1.6
io.projectreactor.netty reactor-netty-http-brave 1.1.6
io.prometheus simpleclient 0.16.0
io.prometheus simpleclient_caffeine 0.16.0
io.prometheus simpleclient_common 0.16.0
io.prometheus simpleclient_dropwizard 0.16.0
io.prometheus simpleclient_graphite_bridge 0.16.0
io.prometheus simpleclient_guava 0.16.0
io.prometheus simpleclient_hibernate 0.16.0
io.prometheus simpleclient_hotspot 0.16.0
io.prometheus simpleclient_httpserver 0.16.0
io.prometheus simpleclient_jetty 0.16.0
io.prometheus simpleclient_jetty_jdk8 0.16.0
io.prometheus simpleclient_log4j 0.16.0
io.prometheus simpleclient_log4j2 0.16.0
io.prometheus simpleclient_logback 0.16.0
io.prometheus simpleclient_pushgateway 0.16.0
io.prometheus simpleclient_servlet 0.16.0
io.prometheus simpleclient_servlet_jakarta 0.16.0
io.prometheus simpleclient_spring_boot 0.16.0
io.prometheus simpleclient_spring_web 0.16.0
io.prometheus simpleclient_tracer_common 0.16.0
io.prometheus simpleclient_tracer_otel 0.16.0
io.prometheus simpleclient_tracer_otel_agent 0.16.0
io.prometheus simpleclient_vertx 0.16.0
io.r2dbc r2dbc-h2 1.0.0.RELEASE
io.r2dbc r2dbc-mssql 1.0.0.RELEASE
io.r2dbc r2dbc-pool 1.0.0.RELEASE
io.r2dbc r2dbc-proxy 1.1.0.RELEASE
io.r2dbc r2dbc-spi 1.0.0.RELEASE
io.reactivex.rxjava3 rxjava 3.1.6
io.rest-assured json-path 5.3.0
io.rest-assured json-schema-validator 5.3.0
io.rest-assured kotlin-extensions 5.3.0
io.rest-assured rest-assured 5.3.0
io.rest-assured rest-assured-all 5.3.0
io.rest-assured rest-assured-common 5.3.0
io.rest-assured scala-support 5.3.0
864
Group ID Artifact ID Version
io.rest-assured spring-commons 5.3.0
io.rest-assured spring-mock-mvc 5.3.0
io.rest-assured spring-mock-mvc-kotlin- 5.3.0
extensions
io.rest-assured spring-web-test-client 5.3.0
io.rest-assured xml-path 5.3.0
io.rsocket rsocket-core 1.1.3
io.rsocket rsocket-load-balancer 1.1.3
io.rsocket rsocket-micrometer 1.1.3
io.rsocket rsocket-test 1.1.3
io.rsocket rsocket-transport-local 1.1.3
io.rsocket rsocket-transport-netty 1.1.3
io.spring.gradle dependency-management-plugin 1.1.0
io.undertow undertow-core 2.3.5.Final
io.undertow undertow-servlet 2.3.5.Final
io.undertow undertow-websockets-jsr 2.3.5.Final
io.zipkin.brave brave 5.15.1
io.zipkin.brave brave-context-jfr 5.15.1
io.zipkin.brave brave-context-log4j12 5.15.1
io.zipkin.brave brave-context-log4j2 5.15.1
io.zipkin.brave brave-context-rxjava2 5.15.1
io.zipkin.brave brave-context-slf4j 5.15.1
io.zipkin.brave brave-instrumentation-dubbo 5.15.1
io.zipkin.brave brave-instrumentation-dubbo- 5.15.1
rpc
io.zipkin.brave brave-instrumentation-grpc 5.15.1
io.zipkin.brave brave-instrumentation-http 5.15.1
io.zipkin.brave brave-instrumentation-http- 5.15.1
tests
io.zipkin.brave brave-instrumentation- 5.15.1
httpasyncclient
io.zipkin.brave brave-instrumentation- 5.15.1
httpclient
io.zipkin.brave brave-instrumentation-jaxrs2 5.15.1
io.zipkin.brave brave-instrumentation-jersey- 5.15.1
server
io.zipkin.brave brave-instrumentation-jms 5.15.1
io.zipkin.brave brave-instrumentation-jms- 5.15.1
jakarta
io.zipkin.brave brave-instrumentation-kafka- 5.15.1
clients
865
Group ID Artifact ID Version
io.zipkin.brave brave-instrumentation-kafka- 5.15.1
streams
io.zipkin.brave brave-instrumentation- 5.15.1
messaging
io.zipkin.brave brave-instrumentation-mongodb 5.15.1
io.zipkin.brave brave-instrumentation-mysql 5.15.1
io.zipkin.brave brave-instrumentation-mysql6 5.15.1
io.zipkin.brave brave-instrumentation-mysql8 5.15.1
io.zipkin.brave brave-instrumentation-netty- 5.15.1
codec-http
io.zipkin.brave brave-instrumentation-okhttp3 5.15.1
io.zipkin.brave brave-instrumentation-p6spy 5.15.1
io.zipkin.brave brave-instrumentation-rpc 5.15.1
io.zipkin.brave brave-instrumentation-servlet 5.15.1
io.zipkin.brave brave-instrumentation-servlet- 5.15.1
jakarta
io.zipkin.brave brave-instrumentation- 5.15.1
sparkjava
io.zipkin.brave brave-instrumentation-spring- 5.15.1
rabbit
io.zipkin.brave brave-instrumentation-spring- 5.15.1
web
io.zipkin.brave brave-instrumentation-spring- 5.15.1
webmvc
io.zipkin.brave brave-instrumentation-vertx- 5.15.1
web
io.zipkin.brave brave-spring-beans 5.15.1
io.zipkin.brave brave-tests 5.15.1
io.zipkin.proto3 zipkin-proto3 1.0.0
io.zipkin.reporter2 zipkin-reporter 2.16.3
io.zipkin.reporter2 zipkin-reporter-brave 2.16.3
io.zipkin.reporter2 zipkin-reporter-metrics- 2.16.3
micrometer
io.zipkin.reporter2 zipkin-reporter-spring-beans 2.16.3
io.zipkin.reporter2 zipkin-sender-activemq-client 2.16.3
io.zipkin.reporter2 zipkin-sender-amqp-client 2.16.3
io.zipkin.reporter2 zipkin-sender-kafka 2.16.3
io.zipkin.reporter2 zipkin-sender-kafka08 2.16.3
io.zipkin.reporter2 zipkin-sender-libthrift 2.16.3
io.zipkin.reporter2 zipkin-sender-okhttp3 2.16.3
io.zipkin.reporter2 zipkin-sender-urlconnection 2.16.3
io.zipkin.zipkin2 zipkin 2.23.2
866
Group ID Artifact ID Version
jakarta.activation jakarta.activation-api 2.1.1
jakarta.annotation jakarta.annotation-api 2.1.1
jakarta.jms jakarta.jms-api 3.1.0
jakarta.json jakarta.json-api 2.1.1
jakarta.json.bind jakarta.json.bind-api 3.0.0
jakarta.mail jakarta.mail-api 2.1.1
jakarta.management.j2ee jakarta.management.j2ee-api 1.1.4
jakarta.persistence jakarta.persistence-api 3.1.0
jakarta.servlet jakarta.servlet-api 6.0.0
jakarta.servlet.jsp.jstl jakarta.servlet.jsp.jstl-api 3.0.0
jakarta.transaction jakarta.transaction-api 2.0.1
jakarta.validation jakarta.validation-api 3.0.2
jakarta.websocket jakarta.websocket-api 2.1.0
jakarta.websocket jakarta.websocket-client-api 2.1.0
jakarta.ws.rs jakarta.ws.rs-api 3.1.0
jakarta.xml.bind jakarta.xml.bind-api 4.0.0
jakarta.xml.soap jakarta.xml.soap-api 3.0.0
jakarta.xml.ws jakarta.xml.ws-api 4.0.0
javax.cache cache-api 1.1.1
javax.money money-api 1.1
jaxen jaxen 2.0.0
junit junit 4.13.2
net.bytebuddy byte-buddy 1.14.4
net.bytebuddy byte-buddy-agent 1.14.4
net.minidev json-smart 2.4.10
net.sourceforge.htmlunit htmlunit 2.70.0
net.sourceforge.jtds jtds 1.3.1
net.sourceforge.nekohtml nekohtml 1.9.22
nz.net.ultraq.thymeleaf thymeleaf-layout-dialect 3.2.1
org.apache.activemq activemq-amqp 5.18.1
org.apache.activemq activemq-blueprint 5.18.1
org.apache.activemq activemq-broker 5.18.1
org.apache.activemq activemq-client 5.18.1
org.apache.activemq activemq-client-jakarta 5.18.1
org.apache.activemq activemq-console 5.18.1
org.apache.activemq activemq-http 5.18.1
org.apache.activemq activemq-jaas 5.18.1
org.apache.activemq activemq-jdbc-store 5.18.1
867
Group ID Artifact ID Version
org.apache.activemq activemq-jms-pool 5.18.1
org.apache.activemq activemq-kahadb-store 5.18.1
org.apache.activemq activemq-karaf 5.18.1
org.apache.activemq activemq-log4j-appender 5.18.1
org.apache.activemq activemq-mqtt 5.18.1
org.apache.activemq activemq-openwire-generator 5.18.1
org.apache.activemq activemq-openwire-legacy 5.18.1
org.apache.activemq activemq-osgi 5.18.1
org.apache.activemq activemq-partition 5.18.1
org.apache.activemq activemq-pool 5.18.1
org.apache.activemq activemq-ra 5.18.1
org.apache.activemq activemq-run 5.18.1
org.apache.activemq activemq-runtime-config 5.18.1
org.apache.activemq activemq-shiro 5.18.1
org.apache.activemq activemq-spring 5.18.1
org.apache.activemq activemq-stomp 5.18.1
org.apache.activemq activemq-web 5.18.1
org.apache.activemq artemis-amqp-protocol 2.28.0
org.apache.activemq artemis-commons 2.28.0
org.apache.activemq artemis-core-client 2.28.0
org.apache.activemq artemis-jakarta-client 2.28.0
org.apache.activemq artemis-jakarta-server 2.28.0
org.apache.activemq artemis-jakarta-service- 2.28.0
extensions
org.apache.activemq artemis-jdbc-store 2.28.0
org.apache.activemq artemis-journal 2.28.0
org.apache.activemq artemis-quorum-api 2.28.0
org.apache.activemq artemis-selector 2.28.0
org.apache.activemq artemis-server 2.28.0
org.apache.activemq artemis-service-extensions 2.28.0
org.apache.commons commons-dbcp2 2.9.0
org.apache.commons commons-lang3 3.12.0
org.apache.commons commons-pool2 2.11.1
org.apache.derby derby 10.16.1.1
org.apache.derby derbyclient 10.16.1.1
org.apache.derby derbynet 10.16.1.1
org.apache.derby derbyoptionaltools 10.16.1.1
org.apache.derby derbyshared 10.16.1.1
org.apache.derby derbytools 10.16.1.1
868
Group ID Artifact ID Version
org.apache.groovy groovy 4.0.11
org.apache.groovy groovy-ant 4.0.11
org.apache.groovy groovy-astbuilder 4.0.11
org.apache.groovy groovy-cli-commons 4.0.11
org.apache.groovy groovy-cli-picocli 4.0.11
org.apache.groovy groovy-console 4.0.11
org.apache.groovy groovy-contracts 4.0.11
org.apache.groovy groovy-datetime 4.0.11
org.apache.groovy groovy-dateutil 4.0.11
org.apache.groovy groovy-docgenerator 4.0.11
org.apache.groovy groovy-ginq 4.0.11
org.apache.groovy groovy-groovydoc 4.0.11
org.apache.groovy groovy-groovysh 4.0.11
org.apache.groovy groovy-jmx 4.0.11
org.apache.groovy groovy-json 4.0.11
org.apache.groovy groovy-jsr223 4.0.11
org.apache.groovy groovy-macro 4.0.11
org.apache.groovy groovy-macro-library 4.0.11
org.apache.groovy groovy-nio 4.0.11
org.apache.groovy groovy-servlet 4.0.11
org.apache.groovy groovy-sql 4.0.11
org.apache.groovy groovy-swing 4.0.11
org.apache.groovy groovy-templates 4.0.11
org.apache.groovy groovy-test 4.0.11
org.apache.groovy groovy-test-junit5 4.0.11
org.apache.groovy groovy-testng 4.0.11
org.apache.groovy groovy-toml 4.0.11
org.apache.groovy groovy-typecheckers 4.0.11
org.apache.groovy groovy-xml 4.0.11
org.apache.groovy groovy-yaml 4.0.11
org.apache.httpcomponents httpasyncclient 4.1.5
org.apache.httpcomponents httpcore 4.4.16
org.apache.httpcomponents httpcore-nio 4.4.16
org.apache.httpcomponents.clie httpclient5 5.2.1
nt5
org.apache.httpcomponents.clie httpclient5-cache 5.2.1
nt5
org.apache.httpcomponents.clie httpclient5-fluent 5.2.1
nt5
869
Group ID Artifact ID Version
org.apache.httpcomponents.clie httpclient5-win 5.2.1
nt5
org.apache.httpcomponents.core httpcore5 5.2.1
5
org.apache.httpcomponents.core httpcore5-h2 5.2.1
5
org.apache.httpcomponents.core httpcore5-reactive 5.2.1
5
org.apache.kafka connect 3.4.0
org.apache.kafka connect-api 3.4.0
org.apache.kafka connect-basic-auth-extension 3.4.0
org.apache.kafka connect-file 3.4.0
org.apache.kafka connect-json 3.4.0
org.apache.kafka connect-mirror 3.4.0
org.apache.kafka connect-mirror-client 3.4.0
org.apache.kafka connect-runtime 3.4.0
org.apache.kafka connect-transforms 3.4.0
org.apache.kafka generator 3.4.0
org.apache.kafka kafka-clients 3.4.0
org.apache.kafka kafka-log4j-appender 3.4.0
org.apache.kafka kafka-metadata 3.4.0
org.apache.kafka kafka-raft 3.4.0
org.apache.kafka kafka-server-common 3.4.0
org.apache.kafka kafka-shell 3.4.0
org.apache.kafka kafka-storage 3.4.0
org.apache.kafka kafka-storage-api 3.4.0
org.apache.kafka kafka-streams 3.4.0
org.apache.kafka kafka-streams-scala_2.12 3.4.0
org.apache.kafka kafka-streams-scala_2.13 3.4.0
org.apache.kafka kafka-streams-test-utils 3.4.0
org.apache.kafka kafka-tools 3.4.0
org.apache.kafka kafka_2.12 3.4.0
org.apache.kafka kafka_2.13 3.4.0
org.apache.kafka trogdor 3.4.0
org.apache.logging.log4j log4j-1.2-api 2.20.0
org.apache.logging.log4j log4j-api 2.20.0
org.apache.logging.log4j log4j-api-test 2.20.0
org.apache.logging.log4j log4j-appserver 2.20.0
org.apache.logging.log4j log4j-cassandra 2.20.0
org.apache.logging.log4j log4j-core 2.20.0
870
Group ID Artifact ID Version
org.apache.logging.log4j log4j-core-test 2.20.0
org.apache.logging.log4j log4j-couchdb 2.20.0
org.apache.logging.log4j log4j-docker 2.20.0
org.apache.logging.log4j log4j-flume-ng 2.20.0
org.apache.logging.log4j log4j-iostreams 2.20.0
org.apache.logging.log4j log4j-jakarta-smtp 2.20.0
org.apache.logging.log4j log4j-jakarta-web 2.20.0
org.apache.logging.log4j log4j-jcl 2.20.0
org.apache.logging.log4j log4j-jmx-gui 2.20.0
org.apache.logging.log4j log4j-jpa 2.20.0
org.apache.logging.log4j log4j-jpl 2.20.0
org.apache.logging.log4j log4j-jul 2.20.0
org.apache.logging.log4j log4j-kubernetes 2.20.0
org.apache.logging.log4j log4j-layout-template-json 2.20.0
org.apache.logging.log4j log4j-layout-template-json- 2.20.0
test
org.apache.logging.log4j log4j-mongodb3 2.20.0
org.apache.logging.log4j log4j-mongodb4 2.20.0
org.apache.logging.log4j log4j-slf4j-impl 2.20.0
org.apache.logging.log4j log4j-slf4j2-impl 2.20.0
org.apache.logging.log4j log4j-spring-boot 2.20.0
org.apache.logging.log4j log4j-spring-cloud-config- 2.20.0
client
org.apache.logging.log4j log4j-taglib 2.20.0
org.apache.logging.log4j log4j-to-jul 2.20.0
org.apache.logging.log4j log4j-to-slf4j 2.20.0
org.apache.logging.log4j log4j-web 2.20.0
org.apache.maven.plugin-tools maven-plugin-annotations 3.6.4
org.apache.tomcat tomcat-annotations-api 10.1.8
org.apache.tomcat tomcat-jdbc 10.1.8
org.apache.tomcat tomcat-jsp-api 10.1.8
org.apache.tomcat.embed tomcat-embed-core 10.1.8
org.apache.tomcat.embed tomcat-embed-el 10.1.8
org.apache.tomcat.embed tomcat-embed-jasper 10.1.8
org.apache.tomcat.embed tomcat-embed-websocket 10.1.8
org.aspectj aspectjrt 1.9.19
org.aspectj aspectjtools 1.9.19
org.aspectj aspectjweaver 1.9.19
org.assertj assertj-core 3.24.2
871
Group ID Artifact ID Version
org.assertj assertj-guava 3.24.2
org.awaitility awaitility 4.2.0
org.awaitility awaitility-groovy 4.2.0
org.awaitility awaitility-kotlin 4.2.0
org.awaitility awaitility-scala 4.2.0
org.cache2k cache2k-api 2.6.1.Final
org.cache2k cache2k-config 2.6.1.Final
org.cache2k cache2k-core 2.6.1.Final
org.cache2k cache2k-jcache 2.6.1.Final
org.cache2k cache2k-micrometer 2.6.1.Final
org.cache2k cache2k-spring 2.6.1.Final
org.codehaus.janino commons-compiler 3.1.9
org.codehaus.janino commons-compiler-jdk 3.1.9
org.codehaus.janino janino 3.1.9
org.eclipse yasson 3.0.3
org.eclipse.angus angus-activation 2.0.0
org.eclipse.angus angus-core 1.1.0
org.eclipse.angus angus-mail 1.1.0
org.eclipse.angus dsn 1.1.0
org.eclipse.angus gimap 1.1.0
org.eclipse.angus imap 1.1.0
org.eclipse.angus jakarta.mail 1.1.0
org.eclipse.angus logging-mailhandler 1.1.0
org.eclipse.angus pop3 1.1.0
org.eclipse.angus smtp 1.1.0
org.eclipse.jetty apache-jsp 11.0.15
org.eclipse.jetty glassfish-jstl 11.0.15
org.eclipse.jetty infinispan-common 11.0.15
org.eclipse.jetty infinispan-embedded-query 11.0.15
org.eclipse.jetty infinispan-remote-query 11.0.15
org.eclipse.jetty jetty-alpn-client 11.0.15
org.eclipse.jetty jetty-alpn-conscrypt-client 11.0.15
org.eclipse.jetty jetty-alpn-conscrypt-server 11.0.15
org.eclipse.jetty jetty-alpn-java-client 11.0.15
org.eclipse.jetty jetty-alpn-java-server 11.0.15
org.eclipse.jetty jetty-alpn-server 11.0.15
org.eclipse.jetty jetty-annotations 11.0.15
org.eclipse.jetty jetty-ant 11.0.15
872
Group ID Artifact ID Version
org.eclipse.jetty jetty-cdi 11.0.15
org.eclipse.jetty jetty-client 11.0.15
org.eclipse.jetty jetty-deploy 11.0.15
org.eclipse.jetty jetty-hazelcast 11.0.15
org.eclipse.jetty jetty-home 11.0.15
org.eclipse.jetty jetty-http 11.0.15
org.eclipse.jetty jetty-http-spi 11.0.15
org.eclipse.jetty jetty-io 11.0.15
org.eclipse.jetty jetty-jaas 11.0.15
org.eclipse.jetty jetty-jaspi 11.0.15
org.eclipse.jetty jetty-jmx 11.0.15
org.eclipse.jetty jetty-jndi 11.0.15
org.eclipse.jetty jetty-keystore 11.0.15
org.eclipse.jetty jetty-nosql 11.0.15
org.eclipse.jetty jetty-openid 11.0.15
org.eclipse.jetty jetty-plus 11.0.15
org.eclipse.jetty jetty-proxy 11.0.15
org.eclipse.jetty jetty-quickstart 11.0.15
org.eclipse.jetty jetty-reactive-httpclient 3.0.8
org.eclipse.jetty jetty-rewrite 11.0.15
org.eclipse.jetty jetty-security 11.0.15
org.eclipse.jetty jetty-server 11.0.15
org.eclipse.jetty jetty-servlet 11.0.15
org.eclipse.jetty jetty-servlets 11.0.15
org.eclipse.jetty jetty-slf4j-impl 11.0.15
org.eclipse.jetty jetty-unixdomain-server 11.0.15
org.eclipse.jetty jetty-unixsocket-client 11.0.15
org.eclipse.jetty jetty-unixsocket-common 11.0.15
org.eclipse.jetty jetty-unixsocket-server 11.0.15
org.eclipse.jetty jetty-util 11.0.15
org.eclipse.jetty jetty-util-ajax 11.0.15
org.eclipse.jetty jetty-webapp 11.0.15
org.eclipse.jetty jetty-xml 11.0.15
org.eclipse.jetty.fcgi fcgi-client 11.0.15
org.eclipse.jetty.fcgi fcgi-server 11.0.15
org.eclipse.jetty.gcloud jetty-gcloud-session-manager 11.0.15
org.eclipse.jetty.http2 http2-client 11.0.15
org.eclipse.jetty.http2 http2-common 11.0.15
873
Group ID Artifact ID Version
org.eclipse.jetty.http2 http2-hpack 11.0.15
org.eclipse.jetty.http2 http2-http-client-transport 11.0.15
org.eclipse.jetty.http2 http2-server 11.0.15
org.eclipse.jetty.http3 http3-client 11.0.15
org.eclipse.jetty.http3 http3-common 11.0.15
org.eclipse.jetty.http3 http3-http-client-transport 11.0.15
org.eclipse.jetty.http3 http3-qpack 11.0.15
org.eclipse.jetty.http3 http3-server 11.0.15
org.eclipse.jetty.memcached jetty-memcached-sessions 11.0.15
org.eclipse.jetty.osgi jetty-httpservice 11.0.15
org.eclipse.jetty.osgi jetty-osgi-alpn 11.0.15
org.eclipse.jetty.osgi jetty-osgi-boot 11.0.15
org.eclipse.jetty.osgi jetty-osgi-boot-jsp 11.0.15
org.eclipse.jetty.osgi jetty-osgi-boot-warurl 11.0.15
org.eclipse.jetty.quic quic-client 11.0.15
org.eclipse.jetty.quic quic-common 11.0.15
org.eclipse.jetty.quic quic-quiche-common 11.0.15
org.eclipse.jetty.quic quic-quiche-jna 11.0.15
org.eclipse.jetty.quic quic-server 11.0.15
org.eclipse.jetty.websocket websocket-core-client 11.0.15
org.eclipse.jetty.websocket websocket-core-common 11.0.15
org.eclipse.jetty.websocket websocket-core-server 11.0.15
org.eclipse.jetty.websocket websocket-jakarta-client 11.0.15
org.eclipse.jetty.websocket websocket-jakarta-common 11.0.15
org.eclipse.jetty.websocket websocket-jakarta-server 11.0.15
org.eclipse.jetty.websocket websocket-jetty-api 11.0.15
org.eclipse.jetty.websocket websocket-jetty-client 11.0.15
org.eclipse.jetty.websocket websocket-jetty-common 11.0.15
org.eclipse.jetty.websocket websocket-jetty-server 11.0.15
org.eclipse.jetty.websocket websocket-servlet 11.0.15
org.ehcache ehcache 3.10.8
org.ehcache ehcache-clustered 3.10.8
org.ehcache ehcache-transactions 3.10.8
org.elasticsearch.client elasticsearch-rest-client 8.7.1
org.elasticsearch.client elasticsearch-rest-client- 8.7.1
sniffer
org.firebirdsql.jdbc jaybird 5.0.1.java11
org.flywaydb flyway-core 9.16.3
org.flywaydb flyway-firebird 9.16.3
874
Group ID Artifact ID Version
org.flywaydb flyway-mysql 9.16.3
org.flywaydb flyway-sqlserver 9.16.3
org.freemarker freemarker 2.3.32
org.glassfish.jaxb codemodel 4.0.2
org.glassfish.jaxb jaxb-core 4.0.2
org.glassfish.jaxb jaxb-jxc 4.0.2
org.glassfish.jaxb jaxb-runtime 4.0.2
org.glassfish.jaxb jaxb-xjc 4.0.2
org.glassfish.jaxb txw2 4.0.2
org.glassfish.jaxb xsom 4.0.2
org.glassfish.jersey.bundles jaxrs-ri 3.1.1
org.glassfish.jersey.connector jersey-apache-connector 3.1.1
s
org.glassfish.jersey.connector jersey-apache5-connector 3.1.1
s
org.glassfish.jersey.connector jersey-grizzly-connector 3.1.1
s
org.glassfish.jersey.connector jersey-helidon-connector 3.1.1
s
org.glassfish.jersey.connector jersey-jdk-connector 3.1.1
s
org.glassfish.jersey.connector jersey-jetty-connector 3.1.1
s
org.glassfish.jersey.connector jersey-jnh-connector 3.1.1
s
org.glassfish.jersey.connector jersey-netty-connector 3.1.1
s
org.glassfish.jersey.container jersey-container-grizzly2-http 3.1.1
s
org.glassfish.jersey.container jersey-container-grizzly2- 3.1.1
s servlet
org.glassfish.jersey.container jersey-container-jdk-http 3.1.1
s
org.glassfish.jersey.container jersey-container-jetty-http 3.1.1
s
org.glassfish.jersey.container jersey-container-jetty-servlet 3.1.1
s
org.glassfish.jersey.container jersey-container-netty-http 3.1.1
s
org.glassfish.jersey.container jersey-container-servlet 3.1.1
s
org.glassfish.jersey.container jersey-container-servlet-core 3.1.1
s
875
Group ID Artifact ID Version
org.glassfish.jersey.container jersey-container-simple-http 3.1.1
s
org.glassfish.jersey.container jersey-gf-ejb 3.1.1
s.glassfish
org.glassfish.jersey.core jersey-client 3.1.1
org.glassfish.jersey.core jersey-common 3.1.1
org.glassfish.jersey.core jersey-server 3.1.1
org.glassfish.jersey.ext jersey-bean-validation 3.1.1
org.glassfish.jersey.ext jersey-declarative-linking 3.1.1
org.glassfish.jersey.ext jersey-entity-filtering 3.1.1
org.glassfish.jersey.ext jersey-metainf-services 3.1.1
org.glassfish.jersey.ext jersey-mvc 3.1.1
org.glassfish.jersey.ext jersey-mvc-bean-validation 3.1.1
org.glassfish.jersey.ext jersey-mvc-freemarker 3.1.1
org.glassfish.jersey.ext jersey-mvc-jsp 3.1.1
org.glassfish.jersey.ext jersey-mvc-mustache 3.1.1
org.glassfish.jersey.ext jersey-proxy-client 3.1.1
org.glassfish.jersey.ext jersey-spring6 3.1.1
org.glassfish.jersey.ext jersey-wadl-doclet 3.1.1
org.glassfish.jersey.ext.cdi jersey-cdi-rs-inject 3.1.1
org.glassfish.jersey.ext.cdi jersey-cdi1x 3.1.1
org.glassfish.jersey.ext.cdi jersey-cdi1x-ban-custom-hk2- 3.1.1
binding
org.glassfish.jersey.ext.cdi jersey-cdi1x-servlet 3.1.1
org.glassfish.jersey.ext.cdi jersey-cdi1x-transaction 3.1.1
org.glassfish.jersey.ext.cdi jersey-cdi1x-validation 3.1.1
org.glassfish.jersey.ext.cdi jersey-weld2-se 3.1.1
org.glassfish.jersey.ext.micro jersey-mp-config 3.1.1
profile
org.glassfish.jersey.ext.micro jersey-mp-rest-client 3.1.1
profile
org.glassfish.jersey.ext.rx jersey-rx-client-guava 3.1.1
org.glassfish.jersey.ext.rx jersey-rx-client-rxjava 3.1.1
org.glassfish.jersey.ext.rx jersey-rx-client-rxjava2 3.1.1
org.glassfish.jersey.inject jersey-cdi2-se 3.1.1
org.glassfish.jersey.inject jersey-hk2 3.1.1
org.glassfish.jersey.media jersey-media-jaxb 3.1.1
org.glassfish.jersey.media jersey-media-json-binding 3.1.1
org.glassfish.jersey.media jersey-media-json-gson 3.1.1
org.glassfish.jersey.media jersey-media-json-jackson 3.1.1
876
Group ID Artifact ID Version
org.glassfish.jersey.media jersey-media-json-jettison 3.1.1
org.glassfish.jersey.media jersey-media-json-processing 3.1.1
org.glassfish.jersey.media jersey-media-kryo 3.1.1
org.glassfish.jersey.media jersey-media-moxy 3.1.1
org.glassfish.jersey.media jersey-media-multipart 3.1.1
org.glassfish.jersey.media jersey-media-sse 3.1.1
org.glassfish.jersey.security oauth1-client 3.1.1
org.glassfish.jersey.security oauth1-server 3.1.1
org.glassfish.jersey.security oauth1-signature 3.1.1
org.glassfish.jersey.security oauth2-client 3.1.1
org.glassfish.jersey.test- jersey-test-framework-core 3.1.1
framework
org.glassfish.jersey.test- jersey-test-framework-util 3.1.1
framework
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-bundle
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-external
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-grizzly2
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-inmemory
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-jdk-http
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-jetty
org.glassfish.jersey.test- jersey-test-framework- 3.1.1
framework.providers provider-simple
org.glassfish.web jakarta.servlet.jsp.jstl 3.0.1
org.hamcrest hamcrest 2.2
org.hamcrest hamcrest-core 2.2
org.hamcrest hamcrest-library 2.2
org.hibernate.orm hibernate-agroal 6.2.2.Final
org.hibernate.orm hibernate-ant 6.2.2.Final
org.hibernate.orm hibernate-c3p0 6.2.2.Final
org.hibernate.orm hibernate-community-dialects 6.2.2.Final
org.hibernate.orm hibernate-core 6.2.2.Final
org.hibernate.orm hibernate-envers 6.2.2.Final
org.hibernate.orm hibernate-graalvm 6.2.2.Final
org.hibernate.orm hibernate-hikaricp 6.2.2.Final
org.hibernate.orm hibernate-jcache 6.2.2.Final
877
Group ID Artifact ID Version
org.hibernate.orm hibernate-jpamodelgen 6.2.2.Final
org.hibernate.orm hibernate-micrometer 6.2.2.Final
org.hibernate.orm hibernate-proxool 6.2.2.Final
org.hibernate.orm hibernate-spatial 6.2.2.Final
org.hibernate.orm hibernate-testing 6.2.2.Final
org.hibernate.orm hibernate-vibur 6.2.2.Final
org.hibernate.validator hibernate-validator 8.0.0.Final
org.hibernate.validator hibernate-validator- 8.0.0.Final
annotation-processor
org.hsqldb hsqldb 2.7.1
org.infinispan infinispan-anchored-keys 14.0.9.Final
org.infinispan infinispan-api 14.0.9.Final
org.infinispan infinispan-cachestore-jdbc 14.0.9.Final
org.infinispan infinispan-cachestore-jdbc- 14.0.9.Final
common
org.infinispan infinispan-cachestore-jdbc- 14.0.9.Final
common-jakarta
org.infinispan infinispan-cachestore-jdbc- 14.0.9.Final
jakarta
org.infinispan infinispan-cachestore-remote 14.0.9.Final
org.infinispan infinispan-cachestore-rocksdb 14.0.9.Final
org.infinispan infinispan-cachestore-sql 14.0.9.Final
org.infinispan infinispan-cdi-common 14.0.9.Final
org.infinispan infinispan-cdi-common-jakarta 14.0.9.Final
org.infinispan infinispan-cdi-embedded 14.0.9.Final
org.infinispan infinispan-cdi-embedded- 14.0.9.Final
jakarta
org.infinispan infinispan-cdi-remote 14.0.9.Final
org.infinispan infinispan-cdi-remote-jakarta 14.0.9.Final
org.infinispan infinispan-checkstyle 14.0.9.Final
org.infinispan infinispan-cli-client 14.0.9.Final
org.infinispan infinispan-client-hotrod 14.0.9.Final
org.infinispan infinispan-client-hotrod- 14.0.9.Final
jakarta
org.infinispan infinispan-client-rest 14.0.9.Final
org.infinispan infinispan-cloudevents- 14.0.9.Final
integration
org.infinispan infinispan-clustered-counter 14.0.9.Final
org.infinispan infinispan-clustered-lock 14.0.9.Final
org.infinispan infinispan-commons 14.0.9.Final
org.infinispan infinispan-commons-jakarta 14.0.9.Final
878
Group ID Artifact ID Version
org.infinispan infinispan-commons-test 14.0.9.Final
org.infinispan infinispan-component- 14.0.9.Final
annotations
org.infinispan infinispan-component-processor 14.0.9.Final
org.infinispan infinispan-console 14.0.8.Final
org.infinispan infinispan-core 14.0.9.Final
org.infinispan infinispan-core-jakarta 14.0.9.Final
org.infinispan infinispan-extended-statistics 14.0.9.Final
org.infinispan infinispan-hibernate-cache- 14.0.9.Final
commons
org.infinispan infinispan-hibernate-cache-spi 14.0.9.Final
org.infinispan infinispan-hibernate-cache-v60 14.0.9.Final
org.infinispan infinispan-hotrod 14.0.9.Final
org.infinispan infinispan-hotrod-jakarta 14.0.9.Final
org.infinispan infinispan-jboss-marshalling 14.0.9.Final
org.infinispan infinispan-jcache 14.0.9.Final
org.infinispan infinispan-jcache-commons 14.0.9.Final
org.infinispan infinispan-jcache-remote 14.0.9.Final
org.infinispan infinispan-key-value-store- 14.0.9.Final
client
org.infinispan infinispan-marshaller-kryo 14.0.9.Final
org.infinispan infinispan-marshaller-kryo- 14.0.9.Final
bundle
org.infinispan infinispan-marshaller- 14.0.9.Final
protostuff
org.infinispan infinispan-marshaller- 14.0.9.Final
protostuff-bundle
org.infinispan infinispan-multimap 14.0.9.Final
org.infinispan infinispan-objectfilter 14.0.9.Final
org.infinispan infinispan-query 14.0.9.Final
org.infinispan infinispan-query-core 14.0.9.Final
org.infinispan infinispan-query-dsl 14.0.9.Final
org.infinispan infinispan-query-jakarta 14.0.9.Final
org.infinispan infinispan-remote-query-client 14.0.9.Final
org.infinispan infinispan-remote-query-server 14.0.9.Final
org.infinispan infinispan-scripting 14.0.9.Final
org.infinispan infinispan-server-core 14.0.9.Final
org.infinispan infinispan-server-hotrod 14.0.9.Final
org.infinispan infinispan-server-hotrod- 14.0.9.Final
jakarta
org.infinispan infinispan-server-memcached 14.0.9.Final
879
Group ID Artifact ID Version
org.infinispan infinispan-server-resp 14.0.9.Final
org.infinispan infinispan-server-rest 14.0.9.Final
org.infinispan infinispan-server-router 14.0.9.Final
org.infinispan infinispan-server-runtime 14.0.9.Final
org.infinispan infinispan-server-testdriver- 14.0.9.Final
core
org.infinispan infinispan-server-testdriver- 14.0.9.Final
junit4
org.infinispan infinispan-server-testdriver- 14.0.9.Final
junit5
org.infinispan infinispan-spring-boot- 14.0.9.Final
starter-embedded
org.infinispan infinispan-spring-boot- 14.0.9.Final
starter-remote
org.infinispan infinispan-spring-boot3- 14.0.9.Final
starter-embedded
org.infinispan infinispan-spring-boot3- 14.0.9.Final
starter-remote
org.infinispan infinispan-spring5-common 14.0.9.Final
org.infinispan infinispan-spring5-embedded 14.0.9.Final
org.infinispan infinispan-spring5-remote 14.0.9.Final
org.infinispan infinispan-spring6-common 14.0.9.Final
org.infinispan infinispan-spring6-embedded 14.0.9.Final
org.infinispan infinispan-spring6-remote 14.0.9.Final
org.infinispan infinispan-tasks 14.0.9.Final
org.infinispan infinispan-tasks-api 14.0.9.Final
org.infinispan infinispan-tools 14.0.9.Final
org.infinispan infinispan-tools-jakarta 14.0.9.Final
org.infinispan.protostream protostream 4.6.2.Final
org.infinispan.protostream protostream-processor 4.6.2.Final
org.infinispan.protostream protostream-types 4.6.2.Final
org.influxdb influxdb-java 2.23
org.jboss.logging jboss-logging 3.5.0.Final
org.jdom jdom2 2.0.6.1
org.jetbrains.kotlin kotlin-compiler 1.8.21
org.jetbrains.kotlin kotlin-compiler-embeddable 1.8.21
org.jetbrains.kotlin kotlin-daemon-client 1.8.21
org.jetbrains.kotlin kotlin-main-kts 1.8.21
org.jetbrains.kotlin kotlin-osgi-bundle 1.8.21
org.jetbrains.kotlin kotlin-reflect 1.8.21
org.jetbrains.kotlin kotlin-script-runtime 1.8.21
880
Group ID Artifact ID Version
org.jetbrains.kotlin kotlin-script-util 1.8.21
org.jetbrains.kotlin kotlin-scripting-common 1.8.21
org.jetbrains.kotlin kotlin-scripting-ide-services 1.8.21
org.jetbrains.kotlin kotlin-scripting-jvm 1.8.21
org.jetbrains.kotlin kotlin-scripting-jvm-host 1.8.21
org.jetbrains.kotlin kotlin-stdlib 1.8.21
org.jetbrains.kotlin kotlin-stdlib-common 1.8.21
org.jetbrains.kotlin kotlin-stdlib-jdk7 1.8.21
org.jetbrains.kotlin kotlin-stdlib-jdk8 1.8.21
org.jetbrains.kotlin kotlin-stdlib-js 1.8.21
org.jetbrains.kotlin kotlin-test 1.8.21
org.jetbrains.kotlin kotlin-test-annotations-common 1.8.21
org.jetbrains.kotlin kotlin-test-common 1.8.21
org.jetbrains.kotlin kotlin-test-js 1.8.21
org.jetbrains.kotlin kotlin-test-junit 1.8.21
org.jetbrains.kotlin kotlin-test-junit5 1.8.21
org.jetbrains.kotlin kotlin-test-testng 1.8.21
org.jetbrains.kotlinx kotlinx-coroutines-android 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-core 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-core-jvm 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-debug 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-guava 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-javafx 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-jdk8 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-jdk9 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-play- 1.6.4
services
org.jetbrains.kotlinx kotlinx-coroutines-reactive 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-reactor 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-rx2 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-rx3 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-slf4j 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-swing 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-test 1.6.4
org.jetbrains.kotlinx kotlinx-coroutines-test-jvm 1.6.4
org.jooq jooq 3.18.3
org.jooq jooq-codegen 3.18.3
org.jooq jooq-kotlin 3.18.3
org.jooq jooq-meta 3.18.3
881
Group ID Artifact ID Version
org.junit.jupiter junit-jupiter 5.9.3
org.junit.jupiter junit-jupiter-api 5.9.3
org.junit.jupiter junit-jupiter-engine 5.9.3
org.junit.jupiter junit-jupiter-migrationsupport 5.9.3
org.junit.jupiter junit-jupiter-params 5.9.3
org.junit.platform junit-platform-commons 1.9.3
org.junit.platform junit-platform-console 1.9.3
org.junit.platform junit-platform-engine 1.9.3
org.junit.platform junit-platform-jfr 1.9.3
org.junit.platform junit-platform-launcher 1.9.3
org.junit.platform junit-platform-reporting 1.9.3
org.junit.platform junit-platform-runner 1.9.3
org.junit.platform junit-platform-suite 1.9.3
org.junit.platform junit-platform-suite-api 1.9.3
org.junit.platform junit-platform-suite-commons 1.9.3
org.junit.platform junit-platform-suite-engine 1.9.3
org.junit.platform junit-platform-testkit 1.9.3
org.junit.vintage junit-vintage-engine 5.9.3
org.jvnet.staxex stax-ex 2.1.0
org.liquibase liquibase-cdi 4.20.0
org.liquibase liquibase-core 4.20.0
org.mariadb r2dbc-mariadb 1.1.4
org.mariadb.jdbc mariadb-java-client 3.1.4
org.messaginghub pooled-jms 3.1.0
org.mockito mockito-android 5.3.1
org.mockito mockito-core 5.3.1
org.mockito mockito-errorprone 5.3.1
org.mockito mockito-junit-jupiter 5.3.1
org.mockito mockito-proxy 5.3.1
org.mockito mockito-subclass 5.3.1
org.mongodb bson 4.9.1
org.mongodb bson-record-codec 4.9.1
org.mongodb mongodb-driver-core 4.9.1
org.mongodb mongodb-driver-legacy 4.9.1
org.mongodb mongodb-driver-reactivestreams 4.9.1
org.mongodb mongodb-driver-sync 4.9.1
org.neo4j.driver neo4j-java-driver 5.7.0
org.postgresql postgresql 42.6.0
882
Group ID Artifact ID Version
org.postgresql r2dbc-postgresql 1.0.1.RELEASE
org.projectlombok lombok 1.18.26
org.quartz-scheduler quartz 2.3.2
org.quartz-scheduler quartz-jobs 2.3.2
org.reactivestreams reactive-streams 1.0.4
org.seleniumhq.selenium htmlunit-driver 4.8.3
org.seleniumhq.selenium lift 4.8.3
org.seleniumhq.selenium selenium-api 4.8.3
org.seleniumhq.selenium selenium-chrome-driver 4.8.3
org.seleniumhq.selenium selenium-chromium-driver 4.8.3
org.seleniumhq.selenium selenium-devtools-v108 4.8.3
org.seleniumhq.selenium selenium-devtools-v109 4.8.3
org.seleniumhq.selenium selenium-devtools-v110 4.8.3
org.seleniumhq.selenium selenium-devtools-v85 4.8.3
org.seleniumhq.selenium selenium-edge-driver 4.8.3
org.seleniumhq.selenium selenium-firefox-driver 4.8.3
org.seleniumhq.selenium selenium-grid 4.8.3
org.seleniumhq.selenium selenium-http 4.8.3
org.seleniumhq.selenium selenium-http-jdk-client 4.8.3
org.seleniumhq.selenium selenium-ie-driver 4.8.3
org.seleniumhq.selenium selenium-java 4.8.3
org.seleniumhq.selenium selenium-json 4.8.3
org.seleniumhq.selenium selenium-manager 4.8.3
org.seleniumhq.selenium selenium-remote-driver 4.8.3
org.seleniumhq.selenium selenium-safari-driver 4.8.3
org.seleniumhq.selenium selenium-session-map-jdbc 4.8.3
org.seleniumhq.selenium selenium-session-map-redis 4.8.3
org.seleniumhq.selenium selenium-support 4.8.3
org.skyscreamer jsonassert 1.5.1
org.slf4j jcl-over-slf4j 2.0.7
org.slf4j jul-to-slf4j 2.0.7
org.slf4j log4j-over-slf4j 2.0.7
org.slf4j slf4j-api 2.0.7
org.slf4j slf4j-ext 2.0.7
org.slf4j slf4j-jdk-platform-logging 2.0.7
org.slf4j slf4j-jdk14 2.0.7
org.slf4j slf4j-log4j12 2.0.7
org.slf4j slf4j-nop 2.0.7
883
Group ID Artifact ID Version
org.slf4j slf4j-reload4j 2.0.7
org.slf4j slf4j-simple 2.0.7
org.springframework spring-aop 6.0.8
org.springframework spring-aspects 6.0.8
org.springframework spring-beans 6.0.8
org.springframework spring-context 6.0.8
org.springframework spring-context-indexer 6.0.8
org.springframework spring-context-support 6.0.8
org.springframework spring-core 6.0.8
org.springframework spring-core-test 6.0.8
org.springframework spring-expression 6.0.8
org.springframework spring-instrument 6.0.8
org.springframework spring-jcl 6.0.8
org.springframework spring-jdbc 6.0.8
org.springframework spring-jms 6.0.8
org.springframework spring-messaging 6.0.8
org.springframework spring-orm 6.0.8
org.springframework spring-oxm 6.0.8
org.springframework spring-r2dbc 6.0.8
org.springframework spring-test 6.0.8
org.springframework spring-tx 6.0.8
org.springframework spring-web 6.0.8
org.springframework spring-webflux 6.0.8
org.springframework spring-webmvc 6.0.8
org.springframework spring-websocket 6.0.8
org.springframework.amqp spring-amqp 3.0.4
org.springframework.amqp spring-rabbit 3.0.4
org.springframework.amqp spring-rabbit-junit 3.0.4
org.springframework.amqp spring-rabbit-stream 3.0.4
org.springframework.amqp spring-rabbit-test 3.0.4
org.springframework.batch spring-batch-core 5.0.1
org.springframework.batch spring-batch-infrastructure 5.0.1
org.springframework.batch spring-batch-integration 5.0.1
org.springframework.batch spring-batch-test 5.0.1
org.springframework.boot spring-boot 3.1.0-RC2
org.springframework.boot spring-boot-actuator 3.1.0-RC2
org.springframework.boot spring-boot-actuator- 3.1.0-RC2
autoconfigure
org.springframework.boot spring-boot-autoconfigure 3.1.0-RC2
884
Group ID Artifact ID Version
org.springframework.boot spring-boot-autoconfigure- 3.1.0-RC2
processor
org.springframework.boot spring-boot-buildpack-platform 3.1.0-RC2
org.springframework.boot spring-boot-configuration- 3.1.0-RC2
metadata
org.springframework.boot spring-boot-configuration- 3.1.0-RC2
processor
org.springframework.boot spring-boot-devtools 3.1.0-RC2
org.springframework.boot spring-boot-docker-compose 3.1.0-RC2
org.springframework.boot spring-boot-jarmode-layertools 3.1.0-RC2
org.springframework.boot spring-boot-loader 3.1.0-RC2
org.springframework.boot spring-boot-loader-tools 3.1.0-RC2
org.springframework.boot spring-boot-properties- 3.1.0-RC2
migrator
org.springframework.boot spring-boot-starter 3.1.0-RC2
org.springframework.boot spring-boot-starter-activemq 3.1.0-RC2
org.springframework.boot spring-boot-starter-actuator 3.1.0-RC2
org.springframework.boot spring-boot-starter-amqp 3.1.0-RC2
org.springframework.boot spring-boot-starter-aop 3.1.0-RC2
org.springframework.boot spring-boot-starter-artemis 3.1.0-RC2
org.springframework.boot spring-boot-starter-batch 3.1.0-RC2
org.springframework.boot spring-boot-starter-cache 3.1.0-RC2
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
cassandra
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
cassandra-reactive
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
couchbase
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
couchbase-reactive
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
elasticsearch
org.springframework.boot spring-boot-starter-data-jdbc 3.1.0-RC2
org.springframework.boot spring-boot-starter-data-jpa 3.1.0-RC2
org.springframework.boot spring-boot-starter-data-ldap 3.1.0-RC2
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
mongodb
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
mongodb-reactive
org.springframework.boot spring-boot-starter-data-neo4j 3.1.0-RC2
org.springframework.boot spring-boot-starter-data-r2dbc 3.1.0-RC2
org.springframework.boot spring-boot-starter-data-redis 3.1.0-RC2
885
Group ID Artifact ID Version
org.springframework.boot spring-boot-starter-data- 3.1.0-RC2
redis-reactive
org.springframework.boot spring-boot-starter-data-rest 3.1.0-RC2
org.springframework.boot spring-boot-starter-freemarker 3.1.0-RC2
org.springframework.boot spring-boot-starter-graphql 3.1.0-RC2
org.springframework.boot spring-boot-starter-groovy- 3.1.0-RC2
templates
org.springframework.boot spring-boot-starter-hateoas 3.1.0-RC2
org.springframework.boot spring-boot-starter- 3.1.0-RC2
integration
org.springframework.boot spring-boot-starter-jdbc 3.1.0-RC2
org.springframework.boot spring-boot-starter-jersey 3.1.0-RC2
org.springframework.boot spring-boot-starter-jetty 3.1.0-RC2
org.springframework.boot spring-boot-starter-jooq 3.1.0-RC2
org.springframework.boot spring-boot-starter-json 3.1.0-RC2
org.springframework.boot spring-boot-starter-log4j2 3.1.0-RC2
org.springframework.boot spring-boot-starter-logging 3.1.0-RC2
org.springframework.boot spring-boot-starter-mail 3.1.0-RC2
org.springframework.boot spring-boot-starter-mustache 3.1.0-RC2
org.springframework.boot spring-boot-starter-oauth2- 3.1.0-RC2
authorization-server
org.springframework.boot spring-boot-starter-oauth2- 3.1.0-RC2
client
org.springframework.boot spring-boot-starter-oauth2- 3.1.0-RC2
resource-server
org.springframework.boot spring-boot-starter-quartz 3.1.0-RC2
org.springframework.boot spring-boot-starter-reactor- 3.1.0-RC2
netty
org.springframework.boot spring-boot-starter-rsocket 3.1.0-RC2
org.springframework.boot spring-boot-starter-security 3.1.0-RC2
org.springframework.boot spring-boot-starter-test 3.1.0-RC2
org.springframework.boot spring-boot-starter-thymeleaf 3.1.0-RC2
org.springframework.boot spring-boot-starter-tomcat 3.1.0-RC2
org.springframework.boot spring-boot-starter-undertow 3.1.0-RC2
org.springframework.boot spring-boot-starter-validation 3.1.0-RC2
org.springframework.boot spring-boot-starter-web 3.1.0-RC2
org.springframework.boot spring-boot-starter-web- 3.1.0-RC2
services
org.springframework.boot spring-boot-starter-webflux 3.1.0-RC2
org.springframework.boot spring-boot-starter-websocket 3.1.0-RC2
org.springframework.boot spring-boot-test 3.1.0-RC2
886
Group ID Artifact ID Version
org.springframework.boot spring-boot-test-autoconfigure 3.1.0-RC2
org.springframework.boot spring-boot-testcontainers 3.1.0-RC2
org.springframework.data spring-data-cassandra 4.1.0-RC1
org.springframework.data spring-data-commons 3.1.0-RC1
org.springframework.data spring-data-couchbase 5.1.0-RC1
org.springframework.data spring-data-elasticsearch 5.1.0-RC1
org.springframework.data spring-data-envers 3.1.0-RC1
org.springframework.data spring-data-jdbc 3.1.0-RC1
org.springframework.data spring-data-jpa 3.1.0-RC1
org.springframework.data spring-data-keyvalue 3.1.0-RC1
org.springframework.data spring-data-ldap 3.1.0-RC1
org.springframework.data spring-data-mongodb 4.1.0-RC1
org.springframework.data spring-data-neo4j 7.1.0-RC1
org.springframework.data spring-data-r2dbc 3.1.0-RC1
org.springframework.data spring-data-redis 3.1.0-RC1
org.springframework.data spring-data-relational 3.1.0-RC1
org.springframework.data spring-data-rest-core 4.1.0-RC1
org.springframework.data spring-data-rest-hal-explorer 4.1.0-RC1
org.springframework.data spring-data-rest-webmvc 4.1.0-RC1
org.springframework.graphql spring-graphql 1.2.0-RC1
org.springframework.graphql spring-graphql-test 1.2.0-RC1
org.springframework.hateoas spring-hateoas 2.1.0-RC1
org.springframework.integratio spring-integration-amqp 6.1.0-RC1
n
org.springframework.integratio spring-integration-camel 6.1.0-RC1
n
org.springframework.integratio spring-integration-cassandra 6.1.0-RC1
n
org.springframework.integratio spring-integration-core 6.1.0-RC1
n
org.springframework.integratio spring-integration-event 6.1.0-RC1
n
org.springframework.integratio spring-integration-feed 6.1.0-RC1
n
org.springframework.integratio spring-integration-file 6.1.0-RC1
n
org.springframework.integratio spring-integration-ftp 6.1.0-RC1
n
org.springframework.integratio spring-integration-graphql 6.1.0-RC1
n
org.springframework.integratio spring-integration-groovy 6.1.0-RC1
n
887
Group ID Artifact ID Version
org.springframework.integratio spring-integration-hazelcast 6.1.0-RC1
n
org.springframework.integratio spring-integration-http 6.1.0-RC1
n
org.springframework.integratio spring-integration-ip 6.1.0-RC1
n
org.springframework.integratio spring-integration-jdbc 6.1.0-RC1
n
org.springframework.integratio spring-integration-jms 6.1.0-RC1
n
org.springframework.integratio spring-integration-jmx 6.1.0-RC1
n
org.springframework.integratio spring-integration-jpa 6.1.0-RC1
n
org.springframework.integratio spring-integration-kafka 6.1.0-RC1
n
org.springframework.integratio spring-integration-mail 6.1.0-RC1
n
org.springframework.integratio spring-integration-mongodb 6.1.0-RC1
n
org.springframework.integratio spring-integration-mqtt 6.1.0-RC1
n
org.springframework.integratio spring-integration-r2dbc 6.1.0-RC1
n
org.springframework.integratio spring-integration-redis 6.1.0-RC1
n
org.springframework.integratio spring-integration-rsocket 6.1.0-RC1
n
org.springframework.integratio spring-integration-scripting 6.1.0-RC1
n
org.springframework.integratio spring-integration-security 6.1.0-RC1
n
org.springframework.integratio spring-integration-sftp 6.1.0-RC1
n
org.springframework.integratio spring-integration-smb 6.1.0-RC1
n
org.springframework.integratio spring-integration-stomp 6.1.0-RC1
n
org.springframework.integratio spring-integration-stream 6.1.0-RC1
n
org.springframework.integratio spring-integration-syslog 6.1.0-RC1
n
org.springframework.integratio spring-integration-test 6.1.0-RC1
n
org.springframework.integratio spring-integration-test- 6.1.0-RC1
n support
888
Group ID Artifact ID Version
org.springframework.integratio spring-integration-webflux 6.1.0-RC1
n
org.springframework.integratio spring-integration-websocket 6.1.0-RC1
n
org.springframework.integratio spring-integration-ws 6.1.0-RC1
n
org.springframework.integratio spring-integration-xml 6.1.0-RC1
n
org.springframework.integratio spring-integration-xmpp 6.1.0-RC1
n
org.springframework.integratio spring-integration-zeromq 6.1.0-RC1
n
org.springframework.integratio spring-integration-zip 6.1.0-RC1
n
org.springframework.integratio spring-integration-zookeeper 6.1.0-RC1
n
org.springframework.kafka spring-kafka 3.0.6
org.springframework.kafka spring-kafka-test 3.0.6
org.springframework.ldap spring-ldap-core 3.1.0-RC1
org.springframework.ldap spring-ldap-ldif-core 3.1.0-RC1
org.springframework.ldap spring-ldap-odm 3.1.0-RC1
org.springframework.ldap spring-ldap-test 3.1.0-RC1
org.springframework.restdocs spring-restdocs-asciidoctor 3.0.0
org.springframework.restdocs spring-restdocs-core 3.0.0
org.springframework.restdocs spring-restdocs-mockmvc 3.0.0
org.springframework.restdocs spring-restdocs-restassured 3.0.0
org.springframework.restdocs spring-restdocs-webtestclient 3.0.0
org.springframework.retry spring-retry 2.0.1
org.springframework.security spring-security-acl 6.1.0-RC1
org.springframework.security spring-security-aspects 6.1.0-RC1
org.springframework.security spring-security-cas 6.1.0-RC1
org.springframework.security spring-security-config 6.1.0-RC1
org.springframework.security spring-security-core 6.1.0-RC1
org.springframework.security spring-security-crypto 6.1.0-RC1
org.springframework.security spring-security-data 6.1.0-RC1
org.springframework.security spring-security-ldap 6.1.0-RC1
org.springframework.security spring-security-messaging 6.1.0-RC1
org.springframework.security spring-security-oauth2- 1.1.0-RC1
authorization-server
org.springframework.security spring-security-oauth2-client 6.1.0-RC1
org.springframework.security spring-security-oauth2-core 6.1.0-RC1
889
Group ID Artifact ID Version
org.springframework.security spring-security-oauth2-jose 6.1.0-RC1
org.springframework.security spring-security-oauth2- 6.1.0-RC1
resource-server
org.springframework.security spring-security-rsocket 6.1.0-RC1
org.springframework.security spring-security-saml2-service- 6.1.0-RC1
provider
org.springframework.security spring-security-taglibs 6.1.0-RC1
org.springframework.security spring-security-test 6.1.0-RC1
org.springframework.security spring-security-web 6.1.0-RC1
org.springframework.session spring-session-core 3.1.0-RC1
org.springframework.session spring-session-data-mongodb 3.1.0-RC1
org.springframework.session spring-session-data-redis 3.1.0-RC1
org.springframework.session spring-session-hazelcast 3.1.0-RC1
org.springframework.session spring-session-jdbc 3.1.0-RC1
org.springframework.ws spring-ws-core 4.0.3
org.springframework.ws spring-ws-security 4.0.3
org.springframework.ws spring-ws-support 4.0.3
org.springframework.ws spring-ws-test 4.0.3
org.springframework.ws spring-xml 4.0.3
org.testcontainers azure 1.18.0
org.testcontainers cassandra 1.18.0
org.testcontainers clickhouse 1.18.0
org.testcontainers cockroachdb 1.18.0
org.testcontainers consul 1.18.0
org.testcontainers couchbase 1.18.0
org.testcontainers cratedb 1.18.0
org.testcontainers database-commons 1.18.0
org.testcontainers db2 1.18.0
org.testcontainers dynalite 1.18.0
org.testcontainers elasticsearch 1.18.0
org.testcontainers gcloud 1.18.0
org.testcontainers hivemq 1.18.0
org.testcontainers influxdb 1.18.0
org.testcontainers jdbc 1.18.0
org.testcontainers junit-jupiter 1.18.0
org.testcontainers k3s 1.18.0
org.testcontainers kafka 1.18.0
org.testcontainers localstack 1.18.0
org.testcontainers mariadb 1.18.0
890
Group ID Artifact ID Version
org.testcontainers mockserver 1.18.0
org.testcontainers mongodb 1.18.0
org.testcontainers mssqlserver 1.18.0
org.testcontainers mysql 1.18.0
org.testcontainers neo4j 1.18.0
org.testcontainers nginx 1.18.0
org.testcontainers oracle-xe 1.18.0
org.testcontainers orientdb 1.18.0
org.testcontainers postgresql 1.18.0
org.testcontainers presto 1.18.0
org.testcontainers pulsar 1.18.0
org.testcontainers questdb 1.18.0
org.testcontainers r2dbc 1.18.0
org.testcontainers rabbitmq 1.18.0
org.testcontainers redpanda 1.18.0
org.testcontainers selenium 1.18.0
org.testcontainers solace 1.18.0
org.testcontainers solr 1.18.0
org.testcontainers spock 1.18.0
org.testcontainers testcontainers 1.18.0
org.testcontainers tidb 1.18.0
org.testcontainers toxiproxy 1.18.0
org.testcontainers trino 1.18.0
org.testcontainers vault 1.18.0
org.testcontainers yugabytedb 1.18.0
org.thymeleaf thymeleaf 3.1.1.RELEASE
org.thymeleaf thymeleaf-spring6 3.1.1.RELEASE
org.thymeleaf.extras thymeleaf-extras- 3.1.1.RELEASE
springsecurity6
org.webjars webjars-locator-core 0.52
org.xerial sqlite-jdbc 3.41.2.1
org.xmlunit xmlunit-assertj 2.9.1
org.xmlunit xmlunit-assertj3 2.9.1
org.xmlunit xmlunit-core 2.9.1
org.xmlunit xmlunit-jakarta-jaxb-impl 2.9.1
org.xmlunit xmlunit-legacy 2.9.1
org.xmlunit xmlunit-matchers 2.9.1
org.xmlunit xmlunit-placeholders 2.9.1
org.yaml snakeyaml 1.33
891
Group ID Artifact ID Version
redis.clients jedis 4.3.2
wsdl4j wsdl4j 1.6.3
The following table provides all properties that can be used to override the versions managed by
Spring Boot. Browse the spring-boot-dependencies build.gradle for a complete list of dependencies.
You can learn how to customize these versions in your application in the Build Tool Plugins
documentation.
892
Library Version Property
Glassfish JSTL glassfish-jstl.version
GraphQL Java graphql-java.version
Groovy groovy.version
Gson gson.version
H2 h2.version
Hamcrest hamcrest.version
Hazelcast hazelcast.version
Hibernate hibernate.version
Hibernate Validator hibernate-validator.version
HikariCP hikaricp.version
HSQLDB hsqldb.version
HtmlUnit htmlunit.version
HttpAsyncClient httpasyncclient.version
HttpClient5 httpclient5.version
HttpCore httpcore.version
HttpCore5 httpcore5.version
Infinispan infinispan.version
InfluxDB Java influxdb-java.version
Jackson Bom jackson-bom.version
Jakarta Activation jakarta-activation.version
Jakarta Annotation jakarta-annotation.version
Jakarta JMS jakarta-jms.version
Jakarta Json jakarta-json.version
Jakarta Json Bind jakarta-json-bind.version
Jakarta Mail jakarta-mail.version
Jakarta Management jakarta-management.version
Jakarta Persistence jakarta-persistence.version
Jakarta Servlet jakarta-servlet.version
Jakarta Servlet JSP JSTL jakarta-servlet-jsp-jstl.version
Jakarta Transaction jakarta-transaction.version
Jakarta Validation jakarta-validation.version
Jakarta WebSocket jakarta-websocket.version
Jakarta WS RS jakarta-ws-rs.version
Jakarta XML Bind jakarta-xml-bind.version
Jakarta XML SOAP jakarta-xml-soap.version
Jakarta XML WS jakarta-xml-ws.version
Janino janino.version
Javax Cache javax-cache.version
893
Library Version Property
Javax Money javax-money.version
Jaxen jaxen.version
Jaybird jaybird.version
JBoss Logging jboss-logging.version
JDOM2 jdom2.version
Jedis jedis.version
Jersey jersey.version
Jetty jetty.version
Jetty Reactive HTTPClient jetty-reactive-httpclient.version
JMustache jmustache.version
jOOQ jooq.version
Json Path json-path.version
Json-smart json-smart.version
JsonAssert jsonassert.version
JTDS jtds.version
JUnit junit.version
JUnit Jupiter junit-jupiter.version
Kafka kafka.version
Kotlin kotlin.version
Kotlin Coroutines kotlin-coroutines.version
Lettuce lettuce.version
Liquibase liquibase.version
Log4j2 log4j2.version
Logback logback.version
Lombok lombok.version
MariaDB mariadb.version
Maven AntRun Plugin maven-antrun-plugin.version
Maven Assembly Plugin maven-assembly-plugin.version
Maven Clean Plugin maven-clean-plugin.version
Maven Compiler Plugin maven-compiler-plugin.version
Maven Dependency Plugin maven-dependency-plugin.version
Maven Deploy Plugin maven-deploy-plugin.version
Maven Enforcer Plugin maven-enforcer-plugin.version
Maven Failsafe Plugin maven-failsafe-plugin.version
Maven Help Plugin maven-help-plugin.version
Maven Install Plugin maven-install-plugin.version
Maven Invoker Plugin maven-invoker-plugin.version
Maven Jar Plugin maven-jar-plugin.version
894
Library Version Property
Maven Javadoc Plugin maven-javadoc-plugin.version
Maven Resources Plugin maven-resources-plugin.version
Maven Shade Plugin maven-shade-plugin.version
Maven Source Plugin maven-source-plugin.version
Maven Surefire Plugin maven-surefire-plugin.version
Maven War Plugin maven-war-plugin.version
Micrometer micrometer.version
Micrometer Tracing micrometer-tracing.version
Mockito mockito.version
MongoDB mongodb.version
MSSQL JDBC mssql-jdbc.version
MySQL mysql.version
Native Build Tools Plugin native-build-tools-plugin.version
NekoHTML nekohtml.version
Neo4j Java Driver neo4j-java-driver.version
Netty netty.version
OkHttp okhttp.version
OpenTelemetry opentelemetry.version
Oracle Database oracle-database.version
Oracle R2DBC oracle-r2dbc.version
Pooled JMS pooled-jms.version
Postgresql postgresql.version
Prometheus Client prometheus-client.version
Quartz quartz.version
QueryDSL querydsl.version
R2DBC H2 r2dbc-h2.version
R2DBC MariaDB r2dbc-mariadb.version
R2DBC MSSQL r2dbc-mssql.version
R2DBC MySQL r2dbc-mysql.version
R2DBC Pool r2dbc-pool.version
R2DBC Postgresql r2dbc-postgresql.version
R2DBC Proxy r2dbc-proxy.version
R2DBC SPI r2dbc-spi.version
Rabbit AMQP Client rabbit-amqp-client.version
Rabbit Stream Client rabbit-stream-client.version
Reactive Streams reactive-streams.version
Reactor Bom reactor-bom.version
REST Assured rest-assured.version
895
Library Version Property
RSocket rsocket.version
RxJava3 rxjava3.version
SAAJ Impl saaj-impl.version
Selenium selenium.version
Selenium HtmlUnit selenium-htmlunit.version
SendGrid sendgrid.version
SLF4J slf4j.version
SnakeYAML snakeyaml.version
Spring AMQP spring-amqp.version
Spring Authorization Server spring-authorization-server.version
Spring Batch spring-batch.version
Spring Data Bom spring-data-bom.version
Spring Framework spring-framework.version
Spring GraphQL spring-graphql.version
Spring HATEOAS spring-hateoas.version
Spring Integration spring-integration.version
Spring Kafka spring-kafka.version
Spring LDAP spring-ldap.version
Spring RESTDocs spring-restdocs.version
Spring Retry spring-retry.version
Spring Security spring-security.version
Spring Session spring-session.version
Spring WS spring-ws.version
SQLite JDBC sqlite-jdbc.version
Testcontainers testcontainers.version
Thymeleaf thymeleaf.version
Thymeleaf Extras Data Attribute thymeleaf-extras-data-attribute.version
Thymeleaf Extras SpringSecurity thymeleaf-extras-springsecurity.version
Thymeleaf Layout Dialect thymeleaf-layout-dialect.version
Tomcat tomcat.version
UnboundID LDAPSDK unboundid-ldapsdk.version
Undertow undertow.version
Versions Maven Plugin versions-maven-plugin.version
WebJars Locator Core webjars-locator-core.version
WSDL4j wsdl4j.version
XML Maven Plugin xml-maven-plugin.version
XmlUnit2 xmlunit2.version
Yasson yasson.version
896