1. Overview
Managing the lifecycle of Spring Boot Application is very important for a production-ready system. The Spring container handles the creation, initialization, and destruction of all the Beans with the help of the ApplicationContext.
The emphasize of this write-up is the destruction phase of the lifecycle. More specifically, we’ll have a look at different ways to shut down a Spring Boot Application.
To learn more about how to set up a project using Spring Boot, check out the Spring Boot Starter article, or go over the Spring Boot Configuration.
2. Shutdown Endpoint
By default, all the endpoints are enabled in Spring Boot Application except /shutdown; this is, naturally, part of the Actuator endpoints.
Here’s the Maven dependency to set up these up:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency>
And, if we want to also set up security support, we need:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-security</artifactId> </dependency>
Lastly, we enable the shutdown endpoint in application.properties file:
management.endpoint.shutdown.enabled=true endpoints.shutdown.enabled=true
To shut down the Spring Boot application, we simply call a POST method like this:
curl -X POST localhost:port/shutdown
And a quick test fragment, using the Spring MVC testing support:
mockMvc.perform(post("/shutdown")).andExpect(status().isOk());
3. Close Application Context
We can also call the close() method directly using the application context:
ConfigurableApplicationContext ctx = new SpringApplicationBuilder(Application.class).web(false).run(); System.out.println("Spring Boot application started"); ctx.getBean(TerminateBean.class); ctx.close();
This destroys all the beans, releases the locks, then closes the bean factory. To verify the application shutdown, we use the Spring’s standard lifecycle callback with @PreDestroy annotation:
@PreDestroy public void onDestroy() throws Exception { System.out.println("Spring Container is destroyed!"); }
Here’s the output after running this example:
Spring Boot application started Closing AnnotationConfigApplicationContext@39b43d60 DefaultLifecycleProcessor - Stopping beans in phase 0 Unregistering JMX-exposed beans on shutdown Spring Container is destroyed!
The important thing here to keep in mind: while closing the application context, the parent context isn’t affected due to separate lifecycles.
4. Exit SpringApplication
SpringApplication registers a shutdown hook with the JVM to make sure the application exits appropriately.
Beans may implement the ExitCodeGenerator interface to return a specific error code:
ConfigurableApplicationContext ctx = new SpringApplicationBuilder(Application.class).web(false).run(); int exitCode = SpringApplication.exit(ctx, new ExitCodeGenerator() { @Override public int getExitCode() { // return the error code return 0; } }); System.exit(exitCode);
The same code with the application of Java 8 lambdas:
SpringApplication.exit(ctx, () -> 0);
After calling the System.exit(exitCode), the program terminates with a 0 return code:
Process finished with exit code 0
5. Kill the App Process
Finally, we can also shut down a Spring Boot Application from outside the application by using a bash script. Our first step for this option is to have the application context write it’s PID into a file:
SpringApplicationBuilder app = new SpringApplicationBuilder(Application.class).web(false); app.build().addListeners(new ApplicationPidFileWriter("./bin/shutdown.pid")); app.run();
Next, create a shutdown.bat file with the following content:
kill $(cat ./bin/shutdown.pid)
The execution of shutdown.bat extracts the Process ID from the shutdown.pid file and uses the kill command to terminate the Boot application.
6. Conclusion
In this quick write-up, we’ve covered few simple methods that can be used to shut down a running Spring Boot Application.
While it’s up to the developer to choose an appropriate a method; all of these methods should be used by design and on purpose.
For example, .exit() is preferred when we need to pass an error code to another environment, say JVM for further actions. Using Application PID gives more flexibility, as we can also start or restart the application with the use of bash script.
Finally, /shutdown is here to make it possible to terminate the applications externally via HTTP. For all the other cases .close() will work perfectly.
As usual, the complete code for this article is available over on the GitHub project.