1 package junit.textui;
2 
3 
4 import java.io.PrintStream;
5 
6 import junit.framework.Test;
7 import junit.framework.TestCase;
8 import junit.framework.TestResult;
9 import junit.framework.TestSuite;
10 import junit.runner.BaseTestRunner;
11 import junit.runner.Version;
12 
13 /**
14  * A command line based tool to run tests.
15  * <pre>
16  * java junit.textui.TestRunner [-wait] TestCaseClass
17  * </pre>
18  *
19  * <p>TestRunner expects the name of a TestCase class as argument.
20  * If this class defines a static <code>suite</code> method it
21  * will be invoked and the returned test is run. Otherwise all
22  * the methods starting with "test" having no arguments are run.</p>
23  *
24  * <p> When the wait command line argument is given TestRunner
25  * waits until the users types RETURN.</p>
26  *
27  * <p>TestRunner prints a trace as the tests are executed followed by a
28  * summary at the end.</p>
29  */
30 public class TestRunner extends BaseTestRunner {
31 	private ResultPrinter fPrinter;
32 
33 	public static final int SUCCESS_EXIT= 0;
34 	public static final int FAILURE_EXIT= 1;
35 	public static final int EXCEPTION_EXIT= 2;
36 
37 	/**
38 	 * Constructs a TestRunner.
39 	 */
TestRunner()40 	public TestRunner() {
41 		this(System.out);
42 	}
43 
44 	/**
45 	 * Constructs a TestRunner using the given stream for all the output
46 	 */
TestRunner(PrintStream writer)47 	public TestRunner(PrintStream writer) {
48 		this(new ResultPrinter(writer));
49 	}
50 
51 	/**
52 	 * Constructs a TestRunner using the given ResultPrinter all the output
53 	 */
TestRunner(ResultPrinter printer)54 	public TestRunner(ResultPrinter printer) {
55 		fPrinter= printer;
56 	}
57 
58 	/**
59 	 * Runs a suite extracted from a TestCase subclass.
60 	 */
run(Class<? extends TestCase> testClass)61 	static public void run(Class<? extends TestCase> testClass) {
62 		run(new TestSuite(testClass));
63 	}
64 
65 	/**
66 	 * Runs a single test and collects its results.
67 	 * This method can be used to start a test run
68 	 * from your program.
69 	 * <pre>
70 	 * public static void main (String[] args) {
71 	 *    test.textui.TestRunner.run(suite());
72 	 * }
73 	 * </pre>
74 	 */
run(Test test)75 	static public TestResult run(Test test) {
76 		TestRunner runner= new TestRunner();
77 		return runner.doRun(test);
78 	}
79 
80 	/**
81 	 * Runs a single test and waits until the user
82 	 * types RETURN.
83 	 */
runAndWait(Test suite)84 	static public void runAndWait(Test suite) {
85 		TestRunner aTestRunner= new TestRunner();
86 		aTestRunner.doRun(suite, true);
87 	}
88 
89 	@Override
testFailed(int status, Test test, Throwable t)90 	public void testFailed(int status, Test test, Throwable t) {
91 	}
92 
93 	@Override
testStarted(String testName)94 	public void testStarted(String testName) {
95 	}
96 
97 	@Override
testEnded(String testName)98 	public void testEnded(String testName) {
99 	}
100 
101 	/**
102 	 * Creates the TestResult to be used for the test run.
103 	 */
createTestResult()104 	protected TestResult createTestResult() {
105 		return new TestResult();
106 	}
107 
doRun(Test test)108 	public TestResult doRun(Test test) {
109 		return doRun(test, false);
110 	}
111 
doRun(Test suite, boolean wait)112 	public TestResult doRun(Test suite, boolean wait) {
113 		TestResult result= createTestResult();
114 		result.addListener(fPrinter);
115 		long startTime= System.currentTimeMillis();
116 		suite.run(result);
117 		long endTime= System.currentTimeMillis();
118 		long runTime= endTime-startTime;
119 		fPrinter.print(result, runTime);
120 
121 		pause(wait);
122 		return result;
123 	}
124 
pause(boolean wait)125 	protected void pause(boolean wait) {
126 		if (!wait) return;
127 		fPrinter.printWaitPrompt();
128 		try {
129 			System.in.read();
130 		}
131 		catch(Exception e) {
132 		}
133 	}
134 
main(String args[])135 	public static void main(String args[]) {
136 		TestRunner aTestRunner= new TestRunner();
137 		try {
138 			TestResult r= aTestRunner.start(args);
139 			if (!r.wasSuccessful())
140 				System.exit(FAILURE_EXIT);
141 			System.exit(SUCCESS_EXIT);
142 		} catch(Exception e) {
143 			System.err.println(e.getMessage());
144 			System.exit(EXCEPTION_EXIT);
145 		}
146 	}
147 
148 	/**
149 	 * Starts a test run. Analyzes the command line arguments and runs the given
150 	 * test suite.
151 	 */
start(String args[])152 	public TestResult start(String args[]) throws Exception {
153 		String testCase= "";
154 		String method= "";
155 		boolean wait= false;
156 
157 		for (int i= 0; i < args.length; i++) {
158 			if (args[i].equals("-wait"))
159 				wait= true;
160 			else if (args[i].equals("-c"))
161 				testCase= extractClassName(args[++i]);
162 			else if (args[i].equals("-m")) {
163 				String arg= args[++i];
164 				int lastIndex= arg.lastIndexOf('.');
165 				testCase= arg.substring(0, lastIndex);
166 				method= arg.substring(lastIndex + 1);
167 			} else if (args[i].equals("-v"))
168 				System.err.println("JUnit " + Version.id() + " by Kent Beck and Erich Gamma");
169 			else
170 				testCase= args[i];
171 		}
172 
173 		if (testCase.equals(""))
174 			throw new Exception("Usage: TestRunner [-wait] testCaseName, where name is the name of the TestCase class");
175 
176 		try {
177 			if (!method.equals(""))
178 				return runSingleMethod(testCase, method, wait);
179 			Test suite= getTest(testCase);
180 			return doRun(suite, wait);
181 		} catch (Exception e) {
182 			throw new Exception("Could not create and run test suite: " + e);
183 		}
184 	}
185 
runSingleMethod(String testCase, String method, boolean wait)186 	protected TestResult runSingleMethod(String testCase, String method, boolean wait) throws Exception {
187 		Class<? extends TestCase> testClass= loadSuiteClass(testCase).asSubclass(TestCase.class);
188 		Test test= TestSuite.createTest(testClass, method);
189 		return doRun(test, wait);
190 	}
191 
192 	@Override
runFailed(String message)193 	protected void runFailed(String message) {
194 		System.err.println(message);
195 		System.exit(FAILURE_EXIT);
196 	}
197 
setPrinter(ResultPrinter printer)198 	public void setPrinter(ResultPrinter printer) {
199 		fPrinter= printer;
200 	}
201 
202 
203 }