For the last 3-4 years, I have been working on Spring Boot and its associations like Spring Cloud, Spring Data and Spring Security. I have always been tempted to use the combination of [Ctrl+C] and [taskkill] for the purpose of killing spring boot applications or processes. It can be automated by a batch script, but usually i was quite lazy to do this myself, as the combination of actions used to serve the purpose! :-) Almost always on the command line, the [Ctrl+C] kills the Spring Boot application process, but the outcome is very different on Eclipse (IDE). The process outlives the [Ctrl+C] termination of spring boot application in the case of the IDE. Anyways, this unreliable and inconsistent outcomes across platforms requires us to find out standard or clean ways to shutdown Spring Boot applications and processes. Please note that this article is oriented toward the windows operating system.
GitHub Link to Clone Repository / Fork Code
https://github.com/sumithpuri/skp-spring-boot-shutdown
1. Terminate the Application, Kill the Process (Command Line) - IDE
The most simple method used by most developers. Click on the [Terminate - Red Square in Eclipse on the Console Tab] button. You will observer that the application immediately shuts down. But infact at the background, the tomcat server is still running on the port that it was started!
You can verify this fact by issuing this command on the command-line
netstat -ao
You will get one such screen - Locate the port on which you had started the Tomcat Server.
Go ahead and locate the PID and kill the task (Windows/MS-DOS)
taskkill /F /PID 14836
SpringApplication springBootapplication = new SpringApplication(SpringBootDockerApplication.class);
springBootapplication.addListeners(new ApplicationPidFileWriter("sbshutdownwin.pid"));
springBootapplication.run();
2. Terminate the Application, Kill the Process (TCPView)
There is one another way to immediately kill the process via the TCPView tool that is made available by Microsoft. It is available at this link to download.
The next simple method is proposed by me. Beginby clicking on the [Terminate - Red Square in Eclipse on the Console Tab] button. You will observer that the application immediately shuts down. But infact at the background, the tomcat server will still be running on the port that it was started!
Use the above TCPView tool to locate the process based on the port on which you started the Tomcat Server/Spring Boot Application. Right Click on the process/row item and click on [Kill Process...] and then again Click on [OK] to confirm the killing of the operating system process.
This is a very convenient way especially if you are a Software Architect or an Engineer who has lots of Microservices to develop/manage/maintain. It will be useful when you are in depths of testing/debugging/fixing.
3. Shutdown using Actuator Endpoint
The most 'cleanest' way among the ones I described until now is the usage of Actuator to shutdown the spring boot applications. You will begin this by making sure to the include the following in your pom.xml
Next make sure that you have enabled the shutdown endpoint via actuator using the properties as shown below.
Once you have to shutdown the application, the actuator endpoint /shutdown will also kill the associated process. This is a very clean way to shutdown the Spring Boot applications. You have to make sure that you invoke the actuator endpoint /shutdown via a [POST] request only.
If Spring Boot is the root context of your spring boot application, then the invocation will look like the following. The above diagram shows the response from the request sent via Postman.
http://localhost:8080/springbootdocker/actuator/shutdown
4. Close the Spring Boot ApplicationContext
The application context needs to be stored when you start/run the spring boot application in your spring boot application class.
public class SpringBootDockerApplication {
// Clean Shutdown Method 2 - Actuator Shutdown Endpoint - Store the Context
// Not a Recommended Way to Use public static - Only for the Demo Purposes..
public static ConfigurableApplicationContext ctx;
public static void main(String[] args) {
// Clean Shutdown Method 2 - Actuator Shutdown Endpoint - Store the Context
ctx = SpringApplication.run(SpringBootDockerApplication.class, args);
}
}
You may then use the following controller method to close the context asynchronously. Also, note that a PreDestroy method has been added to do any cleanup.
// Starting Threads using Runnable is Not Always the Best Way..
// Do Explore Other Integrated Approaches such as Async Servlet
@RequestMapping(value = "/shutdown2", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_VALUE)
public ResponseEntity<Object> rule() {
System.out.println("Entry Thread Id (Debug): " + Thread.currentThread().getName());
Runnable runnable= () -> {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
System.out.println("Thread was Interrupted! Error in Thread Sleep (2 Seconds!)");
}
System.out.println("Callable Thread Id: " + Thread.currentThread().getName());
SpringBootDockerApplication.ctx.close();
};
new Thread(runnable).start();
System.out.println("Exit Thread Id (Debug): " + Thread.currentThread().getName());
return new ResponseEntity<>("Shutdown Requested - Will Shutdown in Next 2 Seconds!", HttpStatus.OK);
}
@PreDestroy
public void requestShutdown2PreDestroy() {
System.out.println("Requested Shutdown (via Context) of the Spring Boot Container");
}
Once the above has been done, built and deployed. You may then close the spring boot using the following browser request.
You will notice that the application has been shutdown when you see the console and also the process has been removed from the operating system processes.
5. Exit using SpringApplication.exit()
The other way is use the exit() method of the SpringApplication class, it will also require to use System.exit() for ending the process.
-
// Clean Shutdown Method 2 - Actuator Shutdown Endpoint - Store the Context
// Not a Recommended Way to Use public static - Only for the Demo Purposes..
private static ConfigurableApplicationContext ctx;
public static void main(String[] args) {
// Clean Shutdown Method 2 - Actuator Shutdown Endpoint - Store the Context
ctx = SpringApplication.run(SpringBootDockerApplication.class, args);
}
public static void exitApplication() {
int staticExitCode = SpringApplication.exit(ctx, new ExitCodeGenerator() {
@Override
public int getExitCode() {
// no errors
return 0;
}
});
System.exit(staticExitCode );
}
The controller will look similar to what we had written previously. Once your application is running and you want to shutdown the application 'cleanly', then you may use the following:
[Reference]
https://www.javadevjournal.com/spring-boot/shutdown-spring-boot-application/
No comments:
Post a Comment