forked from I2P_Developers/i2p.i2p
* Shutdown:
- Cancel our JVM shutdown hook when shutting down - Run a spinner task so shutdown always completes - exit() instead of halt() so other JVM shutdown hooks run - Prevent duplicate wrapper notifier hooks - Notify the wrapper twice, once for stopping and once for stopped
This commit is contained in:
@ -298,7 +298,7 @@ public class ConfigNetHandler extends FormHandler {
|
|||||||
private void hiddenSwitch() {
|
private void hiddenSwitch() {
|
||||||
// Full restart required to generate new keys
|
// Full restart required to generate new keys
|
||||||
// FIXME don't call wrapper if not present, only rekey
|
// FIXME don't call wrapper if not present, only rekey
|
||||||
_context.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerAndRekeyTask(Router.EXIT_GRACEFUL_RESTART));
|
ConfigServiceHandler.registerWrapperNotifier(_context, Router.EXIT_GRACEFUL_RESTART, false);
|
||||||
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@ public class ConfigRestartBean {
|
|||||||
// Normal browsers send value, IE sends button label
|
// Normal browsers send value, IE sends button label
|
||||||
if ("shutdownImmediate".equals(action) || _("Shutdown immediately", ctx).equals(action)) {
|
if ("shutdownImmediate".equals(action) || _("Shutdown immediately", ctx).equals(action)) {
|
||||||
if (ctx.hasWrapper())
|
if (ctx.hasWrapper())
|
||||||
ctx.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerTask(Router.EXIT_HARD));
|
ConfigServiceHandler.registerWrapperNotifier(ctx, Router.EXIT_HARD, false);
|
||||||
//ctx.router().shutdown(Router.EXIT_HARD); // never returns
|
//ctx.router().shutdown(Router.EXIT_HARD); // never returns
|
||||||
ctx.router().shutdownGracefully(Router.EXIT_HARD); // give the UI time to respond
|
ctx.router().shutdownGracefully(Router.EXIT_HARD); // give the UI time to respond
|
||||||
} else if ("cancelShutdown".equals(action) || _("Cancel shutdown", ctx).equals(action) ||
|
} else if ("cancelShutdown".equals(action) || _("Cancel shutdown", ctx).equals(action) ||
|
||||||
@ -39,16 +39,16 @@ public class ConfigRestartBean {
|
|||||||
ctx.router().cancelGracefulShutdown();
|
ctx.router().cancelGracefulShutdown();
|
||||||
} else if ("restartImmediate".equals(action) || _("Restart immediately", ctx).equals(action)) {
|
} else if ("restartImmediate".equals(action) || _("Restart immediately", ctx).equals(action)) {
|
||||||
if (ctx.hasWrapper())
|
if (ctx.hasWrapper())
|
||||||
ctx.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerTask(Router.EXIT_HARD_RESTART));
|
ConfigServiceHandler.registerWrapperNotifier(ctx, Router.EXIT_HARD_RESTART, false);
|
||||||
//ctx.router().shutdown(Router.EXIT_HARD_RESTART); // never returns
|
//ctx.router().shutdown(Router.EXIT_HARD_RESTART); // never returns
|
||||||
ctx.router().shutdownGracefully(Router.EXIT_HARD_RESTART); // give the UI time to respond
|
ctx.router().shutdownGracefully(Router.EXIT_HARD_RESTART); // give the UI time to respond
|
||||||
} else if ("restart".equals(action) || _("Restart", ctx).equals(action)) {
|
} else if ("restart".equals(action) || _("Restart", ctx).equals(action)) {
|
||||||
if (ctx.hasWrapper())
|
if (ctx.hasWrapper())
|
||||||
ctx.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerTask(Router.EXIT_GRACEFUL_RESTART));
|
ConfigServiceHandler.registerWrapperNotifier(ctx, Router.EXIT_GRACEFUL_RESTART, false);
|
||||||
ctx.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
ctx.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
||||||
} else if ("shutdown".equals(action) || _("Shutdown", ctx).equals(action)) {
|
} else if ("shutdown".equals(action) || _("Shutdown", ctx).equals(action)) {
|
||||||
if (ctx.hasWrapper())
|
if (ctx.hasWrapper())
|
||||||
ctx.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerTask(Router.EXIT_GRACEFUL));
|
ConfigServiceHandler.registerWrapperNotifier(ctx, Router.EXIT_GRACEFUL, false);
|
||||||
ctx.router().shutdownGracefully();
|
ctx.router().shutdownGracefully();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ import java.util.List;
|
|||||||
import net.i2p.apps.systray.SysTray;
|
import net.i2p.apps.systray.SysTray;
|
||||||
import net.i2p.apps.systray.UrlLauncher;
|
import net.i2p.apps.systray.UrlLauncher;
|
||||||
import net.i2p.router.Router;
|
import net.i2p.router.Router;
|
||||||
|
import net.i2p.router.RouterContext;
|
||||||
import net.i2p.router.startup.ClientAppConfig;
|
import net.i2p.router.startup.ClientAppConfig;
|
||||||
|
|
||||||
import org.tanukisoftware.wrapper.WrapperManager;
|
import org.tanukisoftware.wrapper.WrapperManager;
|
||||||
@ -17,33 +18,113 @@ import org.tanukisoftware.wrapper.WrapperManager;
|
|||||||
*/
|
*/
|
||||||
public class ConfigServiceHandler extends FormHandler {
|
public class ConfigServiceHandler extends FormHandler {
|
||||||
|
|
||||||
public static class UpdateWrapperManagerTask implements Runnable {
|
/**
|
||||||
private int _exitCode;
|
* Register two shutdown hooks, one to rekey and/or tell the wrapper we are stopping,
|
||||||
public UpdateWrapperManagerTask(int exitCode) {
|
* and a final one to tell the wrapper we are stopped.
|
||||||
_exitCode = exitCode;
|
*
|
||||||
}
|
* @since 0.8.8
|
||||||
public void run() {
|
*/
|
||||||
try {
|
private void registerWrapperNotifier(int code, boolean rekey) {
|
||||||
WrapperManager.signalStopped(_exitCode);
|
registerWrapperNotifier(_context, code, rekey);
|
||||||
} catch (Throwable t) {
|
|
||||||
t.printStackTrace();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register two shutdown hooks, one to rekey and/or tell the wrapper we are stopping,
|
||||||
|
* and a final one to tell the wrapper we are stopped.
|
||||||
|
*
|
||||||
|
* @since 0.8.8
|
||||||
|
*/
|
||||||
|
public static void registerWrapperNotifier(RouterContext ctx, int code, boolean rekey) {
|
||||||
|
Runnable task = new UpdateWrapperOrRekeyTask(rekey, ctx.hasWrapper());
|
||||||
|
ctx.addShutdownTask(task);
|
||||||
|
if (ctx.hasWrapper()) {
|
||||||
|
task = new FinalWrapperTask(code);
|
||||||
|
ctx.addFinalShutdownTask(task);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class UpdateWrapperManagerAndRekeyTask implements Runnable {
|
/**
|
||||||
private int _exitCode;
|
* Rekey and/or tell the wrapper we are stopping,
|
||||||
public UpdateWrapperManagerAndRekeyTask(int exitCode) {
|
*/
|
||||||
_exitCode = exitCode;
|
private static class UpdateWrapperOrRekeyTask implements Runnable {
|
||||||
|
private final boolean _rekey;
|
||||||
|
private final boolean _tellWrapper;
|
||||||
|
private static final int HASHCODE = -123999871;
|
||||||
|
private static final int WAIT = 30*1000;
|
||||||
|
|
||||||
|
public UpdateWrapperOrRekeyTask(boolean rekey, boolean tellWrapper) {
|
||||||
|
_rekey = rekey;
|
||||||
|
_tellWrapper = tellWrapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void run() {
|
public void run() {
|
||||||
try {
|
try {
|
||||||
|
if (_rekey)
|
||||||
ContextHelper.getContext(null).router().killKeys();
|
ContextHelper.getContext(null).router().killKeys();
|
||||||
|
if (_tellWrapper)
|
||||||
|
WrapperManager.signalStopping(WAIT);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
t.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make them all look the same since the hooks are stored in a set
|
||||||
|
* and we don't want dups
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return HASHCODE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make them all look the same since the hooks are stored in a set
|
||||||
|
* and we don't want dups
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
return (o != null) && (o instanceof UpdateWrapperOrRekeyTask);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tell the wrapper we are stopped.
|
||||||
|
*
|
||||||
|
* @since 0.8.8
|
||||||
|
*/
|
||||||
|
private static class FinalWrapperTask implements Runnable {
|
||||||
|
private final int _exitCode;
|
||||||
|
private static final int HASHCODE = 123999871;
|
||||||
|
|
||||||
|
public FinalWrapperTask(int exitCode) {
|
||||||
|
_exitCode = exitCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
WrapperManager.signalStopped(_exitCode);
|
WrapperManager.signalStopped(_exitCode);
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
t.printStackTrace();
|
t.printStackTrace();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make them all look the same since the hooks are stored in a set
|
||||||
|
* and we don't want dups
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return HASHCODE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make them all look the same since the hooks are stored in a set
|
||||||
|
* and we don't want dups
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
return (o != null) && (o instanceof FinalWrapperTask);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -52,12 +133,12 @@ public class ConfigServiceHandler extends FormHandler {
|
|||||||
|
|
||||||
if (_("Shutdown gracefully").equals(_action)) {
|
if (_("Shutdown gracefully").equals(_action)) {
|
||||||
if (_context.hasWrapper())
|
if (_context.hasWrapper())
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerTask(Router.EXIT_GRACEFUL));
|
registerWrapperNotifier(Router.EXIT_GRACEFUL, false);
|
||||||
_context.router().shutdownGracefully();
|
_context.router().shutdownGracefully();
|
||||||
addFormNotice(_("Graceful shutdown initiated"));
|
addFormNotice(_("Graceful shutdown initiated"));
|
||||||
} else if (_("Shutdown immediately").equals(_action)) {
|
} else if (_("Shutdown immediately").equals(_action)) {
|
||||||
if (_context.hasWrapper())
|
if (_context.hasWrapper())
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerTask(Router.EXIT_HARD));
|
registerWrapperNotifier(Router.EXIT_HARD, false);
|
||||||
_context.router().shutdown(Router.EXIT_HARD);
|
_context.router().shutdown(Router.EXIT_HARD);
|
||||||
addFormNotice(_("Shutdown immediately! boom bye bye bad bwoy"));
|
addFormNotice(_("Shutdown immediately! boom bye bye bad bwoy"));
|
||||||
} else if (_("Cancel graceful shutdown").equals(_action)) {
|
} else if (_("Cancel graceful shutdown").equals(_action)) {
|
||||||
@ -66,24 +147,22 @@ public class ConfigServiceHandler extends FormHandler {
|
|||||||
} else if (_("Graceful restart").equals(_action)) {
|
} else if (_("Graceful restart").equals(_action)) {
|
||||||
// should have wrapper if restart button is visible
|
// should have wrapper if restart button is visible
|
||||||
if (_context.hasWrapper())
|
if (_context.hasWrapper())
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerTask(Router.EXIT_GRACEFUL_RESTART));
|
registerWrapperNotifier(Router.EXIT_GRACEFUL_RESTART, false);
|
||||||
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
||||||
addFormNotice(_("Graceful restart requested"));
|
addFormNotice(_("Graceful restart requested"));
|
||||||
} else if (_("Hard restart").equals(_action)) {
|
} else if (_("Hard restart").equals(_action)) {
|
||||||
// should have wrapper if restart button is visible
|
// should have wrapper if restart button is visible
|
||||||
if (_context.hasWrapper())
|
if (_context.hasWrapper())
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerTask(Router.EXIT_HARD_RESTART));
|
registerWrapperNotifier(Router.EXIT_HARD_RESTART, false);
|
||||||
_context.router().shutdown(Router.EXIT_HARD_RESTART);
|
_context.router().shutdown(Router.EXIT_HARD_RESTART);
|
||||||
addFormNotice(_("Hard restart requested"));
|
addFormNotice(_("Hard restart requested"));
|
||||||
} else if (_("Rekey and Restart").equals(_action)) {
|
} else if (_("Rekey and Restart").equals(_action)) {
|
||||||
addFormNotice(_("Rekeying after graceful restart"));
|
addFormNotice(_("Rekeying after graceful restart"));
|
||||||
// FIXME don't call wrapper if not present, only rekey
|
registerWrapperNotifier(Router.EXIT_GRACEFUL_RESTART, true);
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerAndRekeyTask(Router.EXIT_GRACEFUL_RESTART));
|
|
||||||
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
||||||
} else if (_("Rekey and Shutdown").equals(_action)) {
|
} else if (_("Rekey and Shutdown").equals(_action)) {
|
||||||
addFormNotice(_("Rekeying after graceful shutdown"));
|
addFormNotice(_("Rekeying after graceful shutdown"));
|
||||||
// FIXME don't call wrapper if not present, only rekey
|
registerWrapperNotifier(Router.EXIT_GRACEFUL, true);
|
||||||
_context.addShutdownTask(new UpdateWrapperManagerAndRekeyTask(Router.EXIT_GRACEFUL));
|
|
||||||
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL);
|
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL);
|
||||||
} else if (_("Run I2P on startup").equals(_action)) {
|
} else if (_("Run I2P on startup").equals(_action)) {
|
||||||
installService();
|
installService();
|
||||||
|
@ -313,7 +313,7 @@ public class UpdateHandler {
|
|||||||
|
|
||||||
protected void restart() {
|
protected void restart() {
|
||||||
if (_context.hasWrapper())
|
if (_context.hasWrapper())
|
||||||
_context.addShutdownTask(new ConfigServiceHandler.UpdateWrapperManagerTask(Router.EXIT_GRACEFUL_RESTART));
|
ConfigServiceHandler.registerWrapperNotifier(_context, Router.EXIT_GRACEFUL_RESTART, false);
|
||||||
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
_context.router().shutdownGracefully(Router.EXIT_GRACEFUL_RESTART);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,3 +1,11 @@
|
|||||||
|
2011-07-15 zzz
|
||||||
|
* Shutdown:
|
||||||
|
- Cancel our JVM shutdown hook when shutting down
|
||||||
|
- Run a spinner task so shutdown always completes
|
||||||
|
- exit() instead of halt() so other JVM shutdown hooks run
|
||||||
|
- Prevent duplicate wrapper notifier hooks
|
||||||
|
- Notify the wrapper twice, once for stopping and once for stopped
|
||||||
|
|
||||||
2011-07-13 zzz
|
2011-07-13 zzz
|
||||||
* Blocklist:
|
* Blocklist:
|
||||||
- Fix delayed lookup of reason from file
|
- Fix delayed lookup of reason from file
|
||||||
|
@ -375,7 +375,9 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
void runRouter() {
|
void runRouter() {
|
||||||
_isAlive = true;
|
_isAlive = true;
|
||||||
_started = _context.clock().now();
|
_started = _context.clock().now();
|
||||||
Runtime.getRuntime().addShutdownHook(_shutdownHook);
|
try {
|
||||||
|
Runtime.getRuntime().removeShutdownHook(_shutdownHook);
|
||||||
|
} catch (IllegalStateException ise) {}
|
||||||
I2PThread.addOOMEventListener(_oomListener);
|
I2PThread.addOOMEventListener(_oomListener);
|
||||||
|
|
||||||
_context.keyManager().startup();
|
_context.keyManager().startup();
|
||||||
@ -644,6 +646,11 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public void rebuildNewIdentity() {
|
public void rebuildNewIdentity() {
|
||||||
|
if (_shutdownHook != null) {
|
||||||
|
try {
|
||||||
|
Runtime.getRuntime().removeShutdownHook(_shutdownHook);
|
||||||
|
} catch (IllegalStateException ise) {}
|
||||||
|
}
|
||||||
killKeys();
|
killKeys();
|
||||||
for (Runnable task : _context.getShutdownTasks()) {
|
for (Runnable task : _context.getShutdownTasks()) {
|
||||||
if (_log.shouldLog(Log.WARN))
|
if (_log.shouldLog(Log.WARN))
|
||||||
@ -961,6 +968,27 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
}
|
}
|
||||||
******/
|
******/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A non-daemon thread to let
|
||||||
|
* the shutdown task get all the way to the end
|
||||||
|
* @since 0.8.8
|
||||||
|
*/
|
||||||
|
private static class Spinner extends Thread {
|
||||||
|
|
||||||
|
public Spinner() {
|
||||||
|
super();
|
||||||
|
setName("Shutdown Spinner");
|
||||||
|
setDaemon(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
sleep(60*1000);
|
||||||
|
} catch (InterruptedException ie) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static final int EXIT_GRACEFUL = 2;
|
public static final int EXIT_GRACEFUL = 2;
|
||||||
public static final int EXIT_HARD = 3;
|
public static final int EXIT_HARD = 3;
|
||||||
public static final int EXIT_OOM = 10;
|
public static final int EXIT_OOM = 10;
|
||||||
@ -971,6 +999,25 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
* Shutdown with no chance of cancellation
|
* Shutdown with no chance of cancellation
|
||||||
*/
|
*/
|
||||||
public void shutdown(int exitCode) {
|
public void shutdown(int exitCode) {
|
||||||
|
if (_shutdownHook != null) {
|
||||||
|
try {
|
||||||
|
Runtime.getRuntime().removeShutdownHook(_shutdownHook);
|
||||||
|
} catch (IllegalStateException ise) {}
|
||||||
|
}
|
||||||
|
shutdown2(exitCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cancel the JVM runtime hook before calling this.
|
||||||
|
*/
|
||||||
|
private void shutdown2(int exitCode) {
|
||||||
|
// So we can get all the way to the end
|
||||||
|
// No, you can't do Thread.currentThread.setDaemon(false)
|
||||||
|
if (_killVMOnEnd) {
|
||||||
|
try {
|
||||||
|
(new Spinner()).start();
|
||||||
|
} catch (Throwable t) {}
|
||||||
|
}
|
||||||
((RouterClock) _context.clock()).removeShiftListener(this);
|
((RouterClock) _context.clock()).removeShiftListener(this);
|
||||||
_isAlive = false;
|
_isAlive = false;
|
||||||
_context.random().saveSeed();
|
_context.random().saveSeed();
|
||||||
@ -1038,6 +1085,9 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
*/
|
*/
|
||||||
private static final boolean ALLOW_DYNAMIC_KEYS = false;
|
private static final boolean ALLOW_DYNAMIC_KEYS = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cancel the JVM runtime hook before calling this.
|
||||||
|
*/
|
||||||
private void finalShutdown(int exitCode) {
|
private void finalShutdown(int exitCode) {
|
||||||
clearCaches();
|
clearCaches();
|
||||||
_log.log(Log.CRIT, "Shutdown(" + exitCode + ") complete" /* , new Exception("Shutdown") */ );
|
_log.log(Log.CRIT, "Shutdown(" + exitCode + ") complete" /* , new Exception("Shutdown") */ );
|
||||||
@ -1052,9 +1102,9 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
if (RouterContext.getContexts().isEmpty())
|
if (RouterContext.getContexts().isEmpty())
|
||||||
RouterContext.killGlobalContext();
|
RouterContext.killGlobalContext();
|
||||||
|
|
||||||
// Since 0.8.8, mainly for Android
|
// Since 0.8.8, for Android and the wrapper
|
||||||
for (Runnable task : _context.getFinalShutdownTasks()) {
|
for (Runnable task : _context.getFinalShutdownTasks()) {
|
||||||
System.err.println("Running final shutdown task " + task.getClass());
|
//System.err.println("Running final shutdown task " + task.getClass());
|
||||||
try {
|
try {
|
||||||
task.run();
|
task.run();
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
@ -1065,7 +1115,9 @@ public class Router implements RouterClock.ClockShiftListener {
|
|||||||
|
|
||||||
if (_killVMOnEnd) {
|
if (_killVMOnEnd) {
|
||||||
try { Thread.sleep(1000); } catch (InterruptedException ie) {}
|
try { Thread.sleep(1000); } catch (InterruptedException ie) {}
|
||||||
Runtime.getRuntime().halt(exitCode);
|
//Runtime.getRuntime().halt(exitCode);
|
||||||
|
// allow the Runtime shutdown hooks to execute
|
||||||
|
Runtime.getRuntime().exit(exitCode);
|
||||||
} else {
|
} else {
|
||||||
Runtime.getRuntime().gc();
|
Runtime.getRuntime().gc();
|
||||||
}
|
}
|
||||||
@ -1772,20 +1824,25 @@ private static class MarkLiveliness implements SimpleTimer.TimedEvent {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Just for failsafe. Standard shutdown should cancel this.
|
||||||
|
*/
|
||||||
private static class ShutdownHook extends Thread {
|
private static class ShutdownHook extends Thread {
|
||||||
private RouterContext _context;
|
private final RouterContext _context;
|
||||||
private static int __id = 0;
|
private static int __id = 0;
|
||||||
private int _id;
|
private final int _id;
|
||||||
|
|
||||||
public ShutdownHook(RouterContext ctx) {
|
public ShutdownHook(RouterContext ctx) {
|
||||||
_context = ctx;
|
_context = ctx;
|
||||||
_id = ++__id;
|
_id = ++__id;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
setName("Router " + _id + " shutdown");
|
setName("Router " + _id + " shutdown");
|
||||||
Log l = _context.logManager().getLog(Router.class);
|
Log l = _context.logManager().getLog(Router.class);
|
||||||
l.log(Log.CRIT, "Shutting down the router...");
|
l.log(Log.CRIT, "Shutting down the router...");
|
||||||
_context.router().shutdown(Router.EXIT_HARD);
|
_context.router().shutdown2(Router.EXIT_HARD);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ public class RouterVersion {
|
|||||||
/** deprecated */
|
/** deprecated */
|
||||||
public final static String ID = "Monotone";
|
public final static String ID = "Monotone";
|
||||||
public final static String VERSION = CoreVersion.VERSION;
|
public final static String VERSION = CoreVersion.VERSION;
|
||||||
public final static long BUILD = 10;
|
public final static long BUILD = 11;
|
||||||
|
|
||||||
/** for example "-test" */
|
/** for example "-test" */
|
||||||
public final static String EXTRA = "";
|
public final static String EXTRA = "";
|
||||||
|
Reference in New Issue
Block a user