/src/l1j/server/server/Shutdown.java

http://l1j-tw-99nets.googlecode.com/ · Java · 329 lines · 184 code · 31 blank · 114 comment · 22 complexity · a10694db1631de402e66af4fdf67667e MD5 · raw file

  1. /**
  2. * License
  3. * THE WORK (AS DEFINED BELOW) IS PROVIDED UNDER THE TERMS OF THIS
  4. * CREATIVE COMMONS PUBLIC LICENSE ("CCPL" OR "LICENSE").
  5. * THE WORK IS PROTECTED BY COPYRIGHT AND/OR OTHER APPLICABLE LAW.
  6. * ANY USE OF THE WORK OTHER THAN AS AUTHORIZED UNDER THIS LICENSE OR
  7. * COPYRIGHT LAW IS PROHIBITED.
  8. *
  9. * BY EXERCISING ANY RIGHTS TO THE WORK PROVIDED HERE, YOU ACCEPT AND
  10. * AGREE TO BE BOUND BY THE TERMS OF THIS LICENSE. TO THE EXTENT THIS LICENSE
  11. * MAY BE CONSIDERED TO BE A CONTRACT, THE LICENSOR GRANTS YOU THE RIGHTS CONTAINED
  12. * HERE IN CONSIDERATION OF YOUR ACCEPTANCE OF SUCH TERMS AND CONDITIONS.
  13. *
  14. */
  15. package l1j.server.server;
  16. import java.util.logging.Logger;
  17. import l1j.server.server.model.Instance.L1PcInstance;
  18. /**
  19. *
  20. * This class provides the functions for shutting down and restarting the server
  21. * It closes all open clientconnections and saves all data.
  22. *
  23. * @version $Revision: 1.2 $ $Date: 2004/11/18 15:43:30 $
  24. */
  25. public class Shutdown extends Thread {
  26. private static Logger _log = Logger.getLogger(Shutdown.class.getName());
  27. private static Shutdown _instance;
  28. private static Shutdown _counterInstance = null;
  29. private int secondsShut;
  30. private int shutdownMode;
  31. public static final int SIGTERM = 0;
  32. public static final int GM_SHUTDOWN = 1;
  33. public static final int GM_RESTART = 2;
  34. public static final int ABORT = 3;
  35. private static String[] _modeText = { "SIGTERM", "shuting down",
  36. "restarting", "aborting" };
  37. /**
  38. * Default constucter is only used internal to create the shutdown-hook
  39. * instance
  40. *
  41. */
  42. public Shutdown() {
  43. secondsShut = -1;
  44. shutdownMode = SIGTERM;
  45. }
  46. /**
  47. * This creates a countdown instance of Shutdown.
  48. *
  49. * @param seconds
  50. * how many seconds until shutdown
  51. * @param restart
  52. * true is the server shall restart after shutdown
  53. *
  54. */
  55. public Shutdown(int seconds, boolean restart) {
  56. if (seconds < 0) {
  57. seconds = 0;
  58. }
  59. secondsShut = seconds;
  60. if (restart) {
  61. shutdownMode = GM_RESTART;
  62. } else {
  63. shutdownMode = GM_SHUTDOWN;
  64. }
  65. }
  66. /**
  67. * get the shutdown-hook instance the shutdown-hook instance is created by
  68. * the first call of this function, but it has to be registrered externaly.
  69. *
  70. * @return instance of Shutdown, to be used as shutdown hook
  71. */
  72. public static Shutdown getInstance() {
  73. if (_instance == null) {
  74. _instance = new Shutdown();
  75. }
  76. return _instance;
  77. }
  78. /**
  79. * this function is called, when a new thread starts
  80. *
  81. * if this thread is the thread of getInstance, then this is the shutdown
  82. * hook and we save all data and disconnect all clients.
  83. *
  84. * after this thread ends, the server will completely exit
  85. *
  86. * if this is not the thread of getInstance, then this is a countdown
  87. * thread. we start the countdown, and when we finished it, and it was not
  88. * aborted, we tell the shutdown-hook why we call exit, and then call exit
  89. *
  90. * when the exit status of the server is 1, startServer.sh / startServer.bat
  91. * will restart the server.
  92. *
  93. */
  94. @Override
  95. public void run() {
  96. if (this == _instance) {
  97. // last byebye, save all data and quit this server
  98. // logging doesnt work here :(
  99. saveData();
  100. // server will quit, when this function ends.
  101. } else {
  102. // gm shutdown: send warnings and then call exit to start shutdown
  103. // sequence
  104. countdown();
  105. // last point where logging is operational :(
  106. _log.warning("GM shutdown countdown is over. "
  107. + _modeText[shutdownMode] + " NOW!");
  108. switch (shutdownMode) {
  109. case GM_SHUTDOWN:
  110. _instance.setMode(GM_SHUTDOWN);
  111. System.gc();// ?????
  112. System.exit(0);
  113. break;
  114. case GM_RESTART:
  115. _instance.setMode(GM_RESTART);
  116. System.gc(); // ?????
  117. System.exit(1);
  118. break;
  119. }
  120. }
  121. }
  122. /**
  123. * This functions starts a shutdown countdown
  124. *
  125. * @param activeChar
  126. * GM who issued the shutdown command
  127. * @param seconds
  128. * seconds until shutdown
  129. * @param restart
  130. * true if the server will restart after shutdown
  131. */
  132. public void startShutdown(L1PcInstance activeChar, int seconds,
  133. boolean restart) {
  134. Announcements _an = Announcements.getInstance();
  135. _log.warning("GM: " + activeChar.getId() + " issued shutdown command. "
  136. + _modeText[shutdownMode] + " in " + seconds + " seconds!");
  137. _an.announceToAll("Server is " + _modeText[shutdownMode] + " in "
  138. + seconds + " seconds!");
  139. if (_counterInstance != null) {
  140. _counterInstance._abort();
  141. }
  142. // the main instance should only run for shutdown hook, so we start a
  143. // new instance
  144. _counterInstance = new Shutdown(seconds, restart);
  145. GeneralThreadPool.getInstance().execute(_counterInstance);
  146. }
  147. /**
  148. * This function aborts a running countdown
  149. *
  150. * @param activeChar
  151. * GM who issued the abort command
  152. */
  153. public void abort(L1PcInstance activeChar) {
  154. Announcements _an = Announcements.getInstance();
  155. _log.warning("GM: " + activeChar.getName() + " issued shutdown ABORT. ");
  156. _an.announceToAll("Server aborts shutdown and continues normal operation!");
  157. if (_counterInstance != null) {
  158. _counterInstance._abort();
  159. }
  160. }
  161. /**
  162. * set the shutdown mode
  163. *
  164. * @param mode
  165. * what mode shall be set
  166. */
  167. private void setMode(int mode) {
  168. shutdownMode = mode;
  169. }
  170. /**
  171. * set the shutdown mode
  172. *
  173. * @param mode
  174. * what mode shall be set
  175. */
  176. int getMode() {
  177. return shutdownMode;
  178. }
  179. /**
  180. * set shutdown mode to ABORT
  181. *
  182. */
  183. private void _abort() {
  184. shutdownMode = ABORT;
  185. }
  186. /**
  187. * this counts the countdown and reports it to all players countdown is
  188. * aborted if mode changes to ABORT
  189. */
  190. private void countdown() {
  191. Announcements _an = Announcements.getInstance();
  192. try {
  193. while (secondsShut > 0) {
  194. switch (secondsShut) {
  195. case 240:
  196. _an.announceToAll("The server will shutdown in 4 minutes.");
  197. break;
  198. case 180:
  199. _an.announceToAll("The server will shutdown in 3 minutes.");
  200. break;
  201. case 120:
  202. _an.announceToAll("The server will shutdown in 2 minutes.");
  203. break;
  204. case 60:
  205. _an.announceToAll("The server will shutdown in 1 minute.");
  206. break;
  207. case 30:
  208. _an.announceToAll("The server will shutdown in 30 seconds.");
  209. break;
  210. case 10:
  211. _an.announceToAll("The server will shutdown in 10 seconds.");
  212. break;
  213. case 5:
  214. _an.announceToAll("The server will shutdown in 5 seconds.");
  215. break;
  216. case 4:
  217. _an.announceToAll("The server will shutdown in 4 seconds.");
  218. break;
  219. case 3:
  220. _an.announceToAll("The server will shutdown in 3 seconds.");
  221. break;
  222. case 2:
  223. _an.announceToAll("The server will shutdown in 2 seconds.");
  224. break;
  225. case 1:
  226. _an.announceToAll("The server will shutdown in 1 second.");
  227. break;
  228. }
  229. secondsShut--;
  230. int delay = 1000; // milliseconds
  231. Thread.sleep(delay);
  232. if (shutdownMode == ABORT) {
  233. break;
  234. }
  235. }
  236. } catch (InterruptedException e) {
  237. // this will never happen
  238. }
  239. }
  240. /**
  241. * this sends a last byebye, disconnects all players and saves data
  242. *
  243. */
  244. private void saveData() {
  245. Announcements _an = Announcements.getInstance();
  246. switch (shutdownMode) {
  247. case SIGTERM:
  248. System.err.println("SIGTERM received. Shutting down NOW!");
  249. break;
  250. case GM_SHUTDOWN:
  251. System.err.println("GM shutdown received. Shutting down NOW!");
  252. break;
  253. case GM_RESTART:
  254. System.err.println("GM restart received. Restarting NOW!");
  255. break;
  256. }
  257. _an.announceToAll("Server is is " + _modeText[shutdownMode]
  258. + " NOW! bye bye");
  259. // we cannt abort shutdown anymore, so i removed the "if"
  260. GameServer.getInstance().disconnectAllCharacters();
  261. System.err
  262. .println("Data saved. All players disconnected, shutting down.");
  263. try {
  264. int delay = 500;
  265. Thread.sleep(delay);
  266. } catch (InterruptedException e) {
  267. // never happens :p
  268. }
  269. }
  270. public void startTelnetShutdown(String IP, int seconds, boolean restart) {
  271. Announcements _an = Announcements.getInstance();
  272. _log.warning("IP: " + IP + " issued shutdown command. "
  273. + _modeText[shutdownMode] + " in " + seconds + " seconds!");
  274. _an.announceToAll("Server is " + _modeText[shutdownMode] + " in "
  275. + seconds + " seconds!");
  276. if (_counterInstance != null) {
  277. _counterInstance._abort();
  278. }
  279. _counterInstance = new Shutdown(seconds, restart);
  280. GeneralThreadPool.getInstance().execute(_counterInstance);
  281. }
  282. /**
  283. * This function aborts a running countdown
  284. *
  285. * @param IP
  286. * IP Which Issued shutdown command
  287. */
  288. public void Telnetabort(String IP) {
  289. Announcements _an = Announcements.getInstance();
  290. _log.warning("IP: " + IP + " issued shutdown ABORT. "
  291. + _modeText[shutdownMode] + " has been stopped!");
  292. _an.announceToAll("Server aborts " + _modeText[shutdownMode]
  293. + " and continues normal operation!");
  294. if (_counterInstance != null) {
  295. _counterInstance._abort();
  296. }
  297. }
  298. }