170

How can I programmatically shutdown a Spring Boot application without terminating the VM?

In other works, what is the opposite of

new SpringApplication(Main.class).run(args); 
4
  • 1
    Good point! Calling close() on that should do the job. Commented Apr 8, 2014 at 17:34
  • Possible duplicate of How to shutdown a Spring Boot Application in a correct way? Commented Jun 14, 2018 at 19:33
  • @AnandVarkeyPhilips No, it definitely isn't. This one is about an API, the other is about a way for ops to do so. Commented Jun 14, 2018 at 20:14
  • 1
    Okay.. That question link might help others. Do you want me to delete the above comment? Commented Jun 15, 2018 at 5:52

5 Answers 5

151

Closing a SpringApplication basically means closing the underlying ApplicationContext. The SpringApplication#run(String...) method gives you that ApplicationContext as a ConfigurableApplicationContext. You can then close() it yourself.

For example,

@SpringBootApplication public class Example { public static void main(String[] args) { ConfigurableApplicationContext ctx = SpringApplication.run(Example.class, args); // ...determine it's time to shut down... ctx.close(); } } 

Alternatively, you can use the static SpringApplication.exit(ApplicationContext, ExitCodeGenerator...) helper method to do it for you. For example,

@SpringBootApplication public class Example { public static void main(String[] args) { ConfigurableApplicationContext ctx = SpringApplication.run(Example.class, args); // ...determine it's time to stop... int exitCode = SpringApplication.exit(ctx, new ExitCodeGenerator() { @Override public int getExitCode() { // no errors return 0; } }); // or shortened to // int exitCode = SpringApplication.exit(ctx, () -> 0); System.exit(exitCode); } } 
Sign up to request clarification or add additional context in comments.

Comments

146

The simplest way would be to inject the following object where you need to initiate the shutdown

ApplicationShutdownManager.java

import org.springframework.context.ApplicationContext; import org.springframework.boot.SpringApplication; @Component class ApplicationShutdownManager { @Autowired private ApplicationContext appContext; /* * Invoke with `0` to indicate no error or different code to indicate * abnormal exit. es: shutdownManager.initiateShutdown(0); **/ public void initiateShutdown(int returnCode){ SpringApplication.exit(appContext, () -> returnCode); } } 

Comments

47

This works, even done is printed.

 SpringApplication.run(MyApplication.class, args).close(); System.out.println("done"); 

So adding .close() after run()

Explanation:

public ConfigurableApplicationContext run(String... args)

Run the Spring application, creating and refreshing a new ApplicationContext. Parameters:

args - the application arguments (usually passed from a Java main method)

Returns: a running ApplicationContext

and:

void close() Close this application context, releasing all resources and locks that the implementation might hold. This includes destroying all cached singleton beans. Note: Does not invoke close on a parent context; parent contexts have their own, independent lifecycle.

This method can be called multiple times without side effects: Subsequent close calls on an already closed context will be ignored.

So basically, it will not close the parent context, that's why the VM doesn't quit.

7 Comments

Just a reminder this solution works for short lived process like batch, but don't use this on Spring MVC applications. The application just shutdown after booting.
@MichaelCOLL the question is about how to programatically shutdown a spring boot app regardless of type. It also works for Spring MVC
@ACV You're right it works, it works very well. But for an app that must stay up (like Spring MVC app), I think, it's not the good way of doing this. In my case, I've used SpringApplication.exit(appContext, () -> returnCode).
What VM are you referring to in your last line? If you're starting your Spring Boot application with SpringApplication.run(MyApplication.class, args), then there is no parent context. There's only one context, the context created and returned by run, which you then immediately close. @Michael is right. This won't work for programs that need to do anything after the Spring context is initialized, which is most programs.
@Savior JVM. There is a parent context. Here we're talking about how to shut down a Spring boot application. You normally don't shut down web applications this way. So this mechanism is usually used for short lived applications that do something then need to stop. By default, Spring boot will just keep on running even after you finished your batch processing so that's where you would want to use this mechanism.
|
21

This will make sure that the SpringBoot application is closed properly and the resources are released back to the operating system,

@Autowired private ApplicationContext context; @GetMapping("/shutdown-app") public void shutdownApp() { int exitCode = SpringApplication.exit(context, (ExitCodeGenerator) () -> 0); System.exit(exitCode); } 

1 Comment

+! as required the System.exit(exitCode) in my application, otherwise spring boot was restarting
3

In the application you can use SpringApplication. This has a static exit() method that takes two arguments: the ApplicationContext and an ExitCodeGenerator:

i.e. you can declare this method:

@Autowired public void shutDown(ExecutorServiceExitCodeGenerator exitCodeGenerator) { SpringApplication.exit(applicationContext, exitCodeGenerator); } 

Inside the Integration tests you can achieved it by adding @DirtiesContext annotation at class level:

  • @DirtiesContext(classMode=ClassMode.AFTER_CLASS) - The associated ApplicationContext will be marked as dirty after the test class.
  • @DirtiesContext(classMode=ClassMode.AFTER_EACH_TEST_METHOD) - The associated ApplicationContext will be marked as dirty after each test method in the class.

i.e.

@RunWith(SpringJUnit4ClassRunner.class) @SpringBootTest(classes = {Application.class}, webEnvironment= SpringBootTest.WebEnvironment.DEFINED_PORT, properties = {"server.port:0"}) @DirtiesContext(classMode= DirtiesContext.ClassMode.AFTER_CLASS) public class ApplicationIT { ... 

1 Comment

Ok. Where am I supposed to get ExecutorServiceExitCodeGenerator? If it is a bean, can you show the creation snippet code (and from which class it is created)? In which class the method shutDown(ExecutorServiceExitCodeGenerator exitCodeGenerator) should be put?

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.