You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

ExecuteWatchdog.java 7.9 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. /*
  2. * The Apache Software License, Version 1.1
  3. *
  4. * Copyright (c) 2000-2001 The Apache Software Foundation. All rights
  5. * reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions
  9. * are met:
  10. *
  11. * 1. Redistributions of source code must retain the above copyright
  12. * notice, this list of conditions and the following disclaimer.
  13. *
  14. * 2. Redistributions in binary form must reproduce the above copyright
  15. * notice, this list of conditions and the following disclaimer in
  16. * the documentation and/or other materials provided with the
  17. * distribution.
  18. *
  19. * 3. The end-user documentation included with the redistribution, if
  20. * any, must include the following acknowlegement:
  21. * "This product includes software developed by the
  22. * Apache Software Foundation (http://www.apache.org/)."
  23. * Alternately, this acknowlegement may appear in the software itself,
  24. * if and wherever such third-party acknowlegements normally appear.
  25. *
  26. * 4. The names "The Jakarta Project", "Ant", and "Apache Software
  27. * Foundation" must not be used to endorse or promote products derived
  28. * from this software without prior written permission. For written
  29. * permission, please contact apache@apache.org.
  30. *
  31. * 5. Products derived from this software may not be called "Apache"
  32. * nor may "Apache" appear in their names without prior written
  33. * permission of the Apache Group.
  34. *
  35. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36. * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37. * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38. * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39. * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42. * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45. * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46. * SUCH DAMAGE.
  47. * ====================================================================
  48. *
  49. * This software consists of voluntary contributions made by many
  50. * individuals on behalf of the Apache Software Foundation. For more
  51. * information on the Apache Software Foundation, please see
  52. * <http://www.apache.org/>.
  53. */
  54. package org.apache.tools.ant.taskdefs;
  55. import org.apache.tools.ant.BuildException;
  56. /**
  57. * Destroys a process running for too long.
  58. * For example:
  59. * <pre>
  60. * ExecuteWatchdog watchdog = new ExecuteWatchdog(30000);
  61. * Execute exec = new Execute(myloghandler, watchdog);
  62. * exec.setCommandLine(mycmdline);
  63. * int exitvalue = exec.execute();
  64. * if (exitvalue != SUCCESS && watchdog.killedProcess()){
  65. * // it was killed on purpose by the watchdog
  66. * }
  67. * </pre>
  68. * @author thomas.haas@softwired-inc.com
  69. * @author <a href="mailto:sbailliez@imediation.com">Stephane Bailliez</a>
  70. * @see Execute
  71. */
  72. public class ExecuteWatchdog implements Runnable {
  73. /** the process to execute and watch for duration */
  74. private Process process;
  75. /** timeout duration. Once the process running time exceeds this it should be killed */
  76. private int timeout;
  77. /** say whether or not the watchog is currently monitoring a process */
  78. private boolean watch = false;
  79. /** exception that might be thrown during the process execution */
  80. private Exception caught = null;
  81. /** say whether or not the process was killed due to running overtime */
  82. private boolean killedProcess = false;
  83. /**
  84. * Creates a new watchdog with a given timeout.
  85. *
  86. * @param timeout the timeout for the process in milliseconds. It must be greather than 0.
  87. */
  88. public ExecuteWatchdog(int timeout) {
  89. if (timeout < 1) {
  90. throw new IllegalArgumentException("timeout lesser than 1.");
  91. }
  92. this.timeout = timeout;
  93. }
  94. /**
  95. * Watches the given process and terminates it, if it runs for too long.
  96. * All information from the previous run are reset.
  97. * @param process the process to monitor. It cannot be <tt>null</tt>
  98. * @throws IllegalStateException thrown if a process is still being monitored.
  99. */
  100. public synchronized void start(Process process) {
  101. if (process == null) {
  102. throw new NullPointerException("process is null.");
  103. }
  104. if (this.process != null) {
  105. throw new IllegalStateException("Already running.");
  106. }
  107. this.caught = null;
  108. this.killedProcess = false;
  109. this.watch = true;
  110. this.process = process;
  111. final Thread thread = new Thread(this, "WATCHDOG");
  112. thread.setDaemon(true);
  113. thread.start();
  114. }
  115. /**
  116. * Stops the watcher. It will notify all threads possibly waiting on this object.
  117. */
  118. public synchronized void stop() {
  119. watch = false;
  120. notifyAll();
  121. }
  122. /**
  123. * Watches the process and terminates it, if it runs for to long.
  124. */
  125. public synchronized void run() {
  126. try {
  127. // This isn't a Task, don't have a Project object to log.
  128. // project.log("ExecuteWatchdog: timeout = "+timeout+" msec", Project.MSG_VERBOSE);
  129. final long until = System.currentTimeMillis() + timeout;
  130. long now;
  131. while (watch && until > (now = System.currentTimeMillis())) {
  132. try {
  133. wait(until - now);
  134. } catch (InterruptedException e) {}
  135. }
  136. // if we are here, either someone stopped the watchdog,
  137. // we are on timeout and the process must be killed, or
  138. // we are on timeout and the process has already stopped.
  139. try {
  140. // We must check if the process was not stopped
  141. // before being here
  142. process.exitValue();
  143. } catch (IllegalThreadStateException e){
  144. // the process is not terminated, if this is really
  145. // a timeout and not a manual stop then kill it.
  146. if (watch){
  147. killedProcess = true;
  148. process.destroy();
  149. }
  150. }
  151. } catch(Exception e) {
  152. caught = e;
  153. } finally {
  154. cleanUp();
  155. }
  156. }
  157. /**
  158. * reset the monitor flag and the process.
  159. */
  160. protected void cleanUp() {
  161. watch = false;
  162. process = null;
  163. }
  164. /**
  165. * This method will rethrow the exception that was possibly caught during the
  166. * run of the process. It will only remains valid once the process has been
  167. * terminated either by 'error', timeout or manual intervention. Information
  168. * will be discarded once a new process is ran.
  169. * @throws BuildException a wrapped exception over the one that was silently
  170. * swallowed and stored during the process run.
  171. */
  172. public void checkException() throws BuildException {
  173. if (caught != null) {
  174. throw new BuildException("Exception in ExecuteWatchdog.run: "
  175. + caught.getMessage(), caught);
  176. }
  177. }
  178. /**
  179. * Indicates whether or not the watchdog is still monitoring the process.
  180. * @return <tt>true</tt> if the process is still running, otherwise <tt>false</tt>.
  181. */
  182. public boolean isWatching(){
  183. return watch;
  184. }
  185. /**
  186. * Indicates whether the last process run was killed on timeout or not.
  187. * @return <tt>true</tt> if the process was killed otherwise <tt>false</tt>.
  188. */
  189. public boolean killedProcess(){
  190. return killedProcess;
  191. }
  192. }