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.

ExecuteWatchdogTest.java 7.5 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  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.Project;
  56. import java.net.*;
  57. import junit.framework.*;
  58. import java.io.*;
  59. /**
  60. * Simple testcase for the ExecuteWatchdog class.
  61. *
  62. * @author <a href="mailto:sbailliez@imediation.com">Stephane Bailliez</a>
  63. */
  64. public class ExecuteWatchdogTest extends TestCase {
  65. private final static int TIME_OUT = 5000;
  66. private final static String TEST_CLASSPATH = getTestClassPath();
  67. private final static int CLOCK_ERROR=200;
  68. private final static int TIME_OUT_TEST=TIME_OUT-CLOCK_ERROR;
  69. private ExecuteWatchdog watchdog;
  70. public ExecuteWatchdogTest(String name) {
  71. super(name);
  72. }
  73. protected void setUp(){
  74. watchdog = new ExecuteWatchdog(TIME_OUT);
  75. }
  76. /**
  77. * Dangerous method to obtain the classpath for the test. This is
  78. * severely tighted to the build.xml properties.
  79. */
  80. private static String getTestClassPath(){
  81. String classpath = System.getProperty("build.tests");
  82. if (classpath == null) {
  83. System.err.println("WARNING: 'build.tests' property is not available !");
  84. classpath = System.getProperty("java.class.path");
  85. }
  86. // JDK 1.1 needs classes.zip in -classpath argument
  87. if (Project.getJavaVersion() == Project.JAVA_1_1) {
  88. classpath += File.pathSeparator
  89. + System.getProperty("java.home")
  90. + File.separator + "lib"
  91. + File.separator + "classes.zip";
  92. }
  93. return classpath;
  94. }
  95. private Process getProcess(int timetorun) throws Exception {
  96. String[] cmdArray = {
  97. "java", "-classpath", TEST_CLASSPATH,
  98. TimeProcess.class.getName(), String.valueOf(timetorun)
  99. };
  100. //System.out.println("Testing with classpath: " + System.getProperty("java.class.path"));
  101. return Runtime.getRuntime().exec(cmdArray);
  102. }
  103. private String getErrorOutput(Process p) throws Exception {
  104. BufferedReader err = new BufferedReader( new InputStreamReader(p.getErrorStream()) );
  105. StringBuffer buf = new StringBuffer();
  106. String line;
  107. while ( (line = err.readLine()) != null){
  108. buf.append(line);
  109. }
  110. return buf.toString();
  111. }
  112. private int waitForEnd(Process p) throws Exception {
  113. int retcode = p.waitFor();
  114. if (retcode != 0){
  115. String err = getErrorOutput(p);
  116. if (err.length() > 0){
  117. System.err.println("ERROR:");
  118. System.err.println(err);
  119. }
  120. }
  121. return retcode;
  122. }
  123. public void testNoTimeOut() throws Exception {
  124. Process process = getProcess(TIME_OUT/2);
  125. watchdog.start(process);
  126. int retCode = waitForEnd(process);
  127. assertTrue("process should not have been killed", !watchdog.killedProcess());
  128. assertEquals(0, retCode);
  129. }
  130. // test that the watchdog ends the process
  131. public void testTimeOut() throws Exception {
  132. Process process = getProcess(TIME_OUT*2);
  133. long now = System.currentTimeMillis();
  134. watchdog.start(process);
  135. int retCode = process.waitFor();
  136. long elapsed = System.currentTimeMillis() - now;
  137. assertTrue("process should have been killed", watchdog.killedProcess());
  138. // assertTrue("return code is invalid: " + retCode, retCode!=0);
  139. assertTrue("elapse time of "+elapsed+" ms is less than timeout value of "+TIME_OUT_TEST+" ms", elapsed >= TIME_OUT_TEST);
  140. assertTrue("elapse time of "+elapsed+" ms is greater than run value of "+(TIME_OUT*2)+" ms", elapsed < TIME_OUT*2);
  141. }
  142. // test a process that runs and failed
  143. public void testFailed() throws Exception {
  144. Process process = getProcess(-1); // process should abort
  145. watchdog.start(process);
  146. int retCode = process.waitFor();
  147. assertTrue("process should not have been killed", !watchdog.killedProcess());
  148. assertTrue("return code is invalid: " + retCode, retCode!=0);
  149. }
  150. public void testManualStop() throws Exception {
  151. final Process process = getProcess(TIME_OUT*2);
  152. watchdog.start(process);
  153. // I assume that starting this takes less than TIME_OUT/2 ms...
  154. Thread thread = new Thread(){
  155. public void run(){
  156. try {
  157. process.waitFor();
  158. } catch(InterruptedException e){
  159. // not very nice but will do the job
  160. fail("process interrupted in thread");
  161. }
  162. }
  163. };
  164. thread.start();
  165. // wait for TIME_OUT/2, there should be about TIME_OUT/2 ms remaining before timeout
  166. thread.join(TIME_OUT/2);
  167. // now stop the watchdog.
  168. watchdog.stop();
  169. // wait for the thread to die, should be the end of the process
  170. thread.join();
  171. // process should be dead and well finished
  172. assertEquals(0, process.exitValue());
  173. assertTrue("process should not have been killed", !watchdog.killedProcess());
  174. }
  175. public static class TimeProcess {
  176. public static void main(String[] args) throws Exception {
  177. int time = Integer.parseInt(args[0]);
  178. if (time < 1) {
  179. throw new IllegalArgumentException("Invalid time: " + time);
  180. }
  181. Thread.sleep(time);
  182. }
  183. }
  184. }