jtreg FAQ
This FAQ is a growing list of questions asked by developers writing tests which will run using the regression extensions to the JavaTest harness, version 3.2.2. It is a supplement to the test-tag language specification and is intended to illuminate implications of the spec and to answer questions about this implementation of the spec.
Index
- Overview
- What's the purpose of this test framework?
- What's a regression test?
- But I can also write non-regression tests using this framework?
- What is the JavaTest harness?
- How do I distinguish the new and improved JavaTest Harness, version 3.* from its predecessors?
- What are the JDK regression extensions to the JavaTest harness? What is "regtest"?
- What is
jtreg? - What are the system requirements for using the JDK regression extensions?
- Where can I find a copy of the JavaTest harness and the regression extensions?
- Where do I find additional supporting documentation?
- There's functionality missing from the tag specification. I can't write my test or it would vastly improve the life of people writing tests if it was added. What do I need to do?
- The spec is fine, but there's some functionality that I'd like to get from the regression extensions or I still can't run it on a basic test. Who do I contact?
- Why not use JUnit?
- Writing a JDK Regression Test
- How do I write a test?
- What does the
@testtag mean? - What do the other tags mean?
- How are tag arguments delimited?
- If a test fails, do I have to throw my own exception?
- Should a test call the
System.exitmethod? - Can a test write to
System.outandSystem.err? - Can a test check the output of
System.outandSystem.err? - What should I do with a test once I've written it?
- Where is the test directory?
- Why not have separate test workspaces that only contain tests?
- How are the test directories organized?
- What should I do if my test directory is missing a subdirectory that I need?
- How should I name a test?
- What about tests that don't fit into the API structure?
- Can tests in different directories have the same name?
- How do I write a test for an AWT bug or a Swing bug?
- How does the user know what to do for a manual applet test?
- Exactly what does the
/manualoption mean? - How does a manual applet test indicate success or failure?
- What if the bug I'm fixing requires a more complex regression test?
- Getting Started
- What does a typical invocation of
jtreglook like? How can I make sure that I can even run the JavaTest harness? - Bleah! That verbose output is so long! Can I have something shorter?
- Are there abbreviations for these long options?
- How do I view what a test sends to
System.out/System.err? - What is a
.jtrfile? - What are the report and work directories?
- What's the difference between the "fail" and "error" return status?
- If a test fails, I'd like to put all of my debugging information into the final result. How do I do that?
- I've heard that the
jtreghas a GUI. How do I access that? - Can I verify the correctness of test descriptions without actually running the tests?
- Can I generate reports for tests that have already been run?
- How do I run
jtregunder Windows? - Which should I use? MKS or Cygwin?
- How do I run only tests which were written for a specific bugid?
- How do I run only tests NOT requiring manual intervention?
- How do I run only tests requiring manual intervention?
- How do I specify VM options to the test's JVM?
- What does a typical invocation of
- General Problems
- My test only passes if I don't use jtreg to run it. Why does it fail in jtreg?
- How do I set the
CLASSPATHenvironment variable for my test? - Why don't you just pass all of my shell environment variables to the JVM running the test?
- Why is the default to run tests in another JVM?
- Why would I ever want to run in the same JVM?
- Should a test call the
System.exitmethod? - My test only applies to one platform and it will fail/not run in others. How do I prevent the harness from running it on the wrong platform?
- How can I make
appletandmainaction tests read from data files? - Can I use
packagestatements in my tests? - Why can't multiple test source files in the same directory have package-private classes of the same name?
- Should a test catch
Throwable,Exception, orError? - My test requires that I use information
printed to
System.outorSystem.errto determine whether a test passed or failed. When I run my test in the harness, I can't seem to find these output streams. - My test does tricky things that are not
supported by
jtreg. Can I still write a regression test? - What happens if my test returns when there are still threads running?
- If my bug hasn't been fixed, and the test is run, the JVM crashes. How do I make sure that the test doesn't cause the harness to crash?
- The JavaTest harness is running into problems running the test because of issues with the JDK I'm trying to test. What can I do?
- My test requires that I install my own security manager, but it appears that the JavaTest harness has already installed one. What do I do?
- Can I automate running regtests or can I run the tests on a regular basis?
- I run all (or a huge part) of the
regression test suite as part of a cron job or other nightly
process. I'd like to generate my own reports or I'd like to send
myself e-mail whenever a test fails. Do I have to parse the verbose
output or the
.jtrfile?
- Tag Problems
- How do I decide whether my test should
use the
compileaction or thebuildaction? - When do I need to specify the
buildaction? - How do I decide whether my applet test
should use the
mainaction or theappletaction? - I put in an
ignoretag into my test description but my test wasn't ignored. - Can I use the
@author,@run, etc. tags in other files?
- How do I decide whether my test should
use the
- Applet Problems
- My
/manualtest sends events toSystem.out/System.errso that the user can determine whether the test behaved properly. How do I write my test if I can't see these output streams? - I threw an exception, the output was
sent to
System.err, but my test still passed. What happened? - My
appletaction test didn't run mymainmethod! - If I have an applet test, do I put the
test description in the
.htmlfile or the.javafile? - For my
/manualtests, how do I provide the user instructions to run the test? - For
/manualtests, how is the initial size of the running applet determined?
- My
- Deciphering Common Harness Errors
Failed. Unexpected exit from testError. Can't find `main' methodError. Parse Exception: No class provided for `main'Error. Parse Exception: `applet' requires exactly one file argumentError. Parse Exception: `archive' not supported in file: …test results:no tests selectedTest does not have unique name within work directory
1. Overview
1.1. What's the purpose of this test framework?
The test framework described here is intended primarily for unit and regression tests. It can also be used for functional tests, and even simple product tests -- in other words, just about any type of test except a conformance test. (Conformance tests belong in a Technology Compatibility Kit (TCK), such as the Java Compatibility Kit (JCK)). Tests can often be written as small standalone Java programs, although in some cases an applet or a shell-script might be required.
1.2. What's a regression test?
A regression test is a test written specifically to check that a bug has been fixed and remains fixed. A regression test should fail when run against a build with the bug in question, and pass when run against a build in which the bug has been fixed.
1.3. But I can also write non-regression tests using this framework?
Yes. Suppose, for example, that you evaluate a bug that turns out not to be a bug, but in the course of doing the evaluation you write a test that checks for correct behavior. You can help improve the quality of the JDK by checking this test into the test directory.
1.4. What is the JavaTest harness?
The JavaTest harness is a set of tools designed to execute test programs. It was originally designed to execute tests in the Java 2 SE Technology Compatibility Kit (TCK) (formerly the Java Compatibility Kit (JCK)). Among other things, the harness has evolved the ability to execute non-JCK testsuites. The JDK regression test suite is one such suite.
The JavaTest harness, version 3.2.2 provides a substantially
upgraded GUI, and other internal improvements, and is the base for
developing the next version of regtest.
1.5. How do I distinguish the new and improved JavaTest Harness, version 3.* from its predecessors?
The JavaTest harness, version 3.* is visually distinguished from its predecessors by its use of an "explorer-like" GUI, providing a tree browser on the left and a detail pane on the right.
Henceforth, in this document, the JavaTest harness, version 3.2.2 will be referred to as "the JavaTest harness".
1.6. What are the JDK regression extensions to the JavaTest harness? What is "regtest"?
For the harness to execute tests in a given test suite, it needs
specialized code which knows how to find test descriptions and how
to interpret those descriptions. The JDK
Test Framework: Tag Language Specification provides the needed
descriptions. regtest refers to extensions for the
JavaTest harness that implement this specification.
1.7. What is
jtreg?
Originally, jtreg referred to a custom shell script
that was used to invoke a custom entry point to the JavaTest
harness. The script is no longer required, and the name
jtreg is simply a popular equivalent for the name
regtest.
1.8. What are the system requirements for using the JDK regression extensions?
It is recommended that you run jtreg on a platform that has been certified as Java Compatible. It requires a version equivalent to JDK 1.5.0 or later.
For Windows systems, the regression extensions require the installation of the MKS Toolkit, version 6.1 or greater, or Cygwin.
1.9. Where can I find a copy of the JavaTest harness and the regression extensions?
You can find them under 'jtreg' on the OpenJDK web page.
1.10. Where do I find additional supporting documentation?
Beyond the Java Platform documentation, the following are relevant documentation resources.
- JDK Test Framework: Tag Language Specification - The definitive document defining the test description tags (syntax and behavior).
- The
-helpoption tojtregoffers brief documentation for the complete set of currently available options. - The JavaTest harness provides detailed online help via the
Helpmenu describing the graphical user interface.
1.11. There's functionality missing from the tag specification. I can't write my test or it would vastly improve the life of people writing tests if it was added. What do I need to do?
See the OpenJDK page for a suitable forum or mailing list.
1.12. The spec is fine, but there's some functionality that I'd like to get from the regression extensions or I still can't run it on a basic test. Who do I contact?
Send email to jtreg-comments@sun.com.
1.13. Why not use JUnit?
JUnit was not around when we started writing tests for JDK. And, the test tag specification has been specifically designed for testing JDK, with support for testing applets, command-line interfaces, and so on, as well as simple API tests.
And by now, there are many thousands of tests written for jtreg, so it would not be practical to convert to JUnit.
2. Writing a JDK Regression Test
2.1. How do I write a test?
The simplest test is an ordinary Java program with the usual static main method. If the test fails, it should throw an exception; if it succeeds, it should return normally.
Here's an example:
-
/* @test 1.1 97/10/12 @bug 1234567 @summary Make sure that 1 != 0 */ public class OneZero { public static void main(String[] args) throws Exception { if (1 == 0) { throw new Exception("1 == 0"); } } }
This test could be run on its own with the command
-
<build>/bin/java OneZero
2.2. What does the
@test tag mean?
The @test tag identifies a source file that defines
a test. the harness will automatically run any .java,
.sh, and .html file containing an
@test tag within the appropriate comment; it ignores
any file not containing such a tag or not utilizing one of the
expected extensions.
If necessary the harness will compile the source file, if the
class files are older than the corresponding source files. Other
files which the test depends on must be specified with the
@run build action.
The arguments to the @test tag are ignored by the
harness. For identification it's useful to put the usual SCCS ID
keywords (I and E, each letter surrounded by %) after
the @test tag.
While not part of the tag specification, some tests use the
string "/nodynamiccopyright" after @test
to indicate that that the file should not be subject to automated
copyright processing that might affect the operation of the test,
for example, by affecting the line numbers of the test source
code.
2.3. What do the other tags mean?
The other tags shown above are optional.
The @bug tag should be followed by one or more bug
numbers, separated by spaces. The bug number is useful in
diagnosing test failures. It's OK to write tests that don't have
bug numbers, but if you're writing a test for a specific bug please
include its number in an @bug tag.
The @summary tag describes the condition that is
checked by the test. It is especially useful for non-regression
tests, which by definition don't have bug numbers, but even if
there's a bug number it's helpful to include a summary. Note that a
test summary is generally not the same thing as a Bugtraq synopsis,
since the latter describes the bug rather than the condition that
the bug violates.
2.4. How are tag arguments delimited?
The arguments of a tag are the words between that tag and the next tag, if there is one, or the end of the comment enclosing the tags.
2.5. If a test fails, do I have to throw my own exception?
No. In cases like the above example in which you must check a
condition in order to decide whether the test is to pass or fail,
you have no choice but to construct an exception.
RuntimeException is a convenient choice since it's
unchecked, so you don't have to sprinkle your code with throws
clauses.
On the other hand, if the test would naturally cause an exception to be thrown when it fails, it suffices to let that exception be propagated up through the main method. If the exception you expect to be thrown in the failure case is a checked exception, then you'll need to provide the appropriate throws clauses in your code.
In general, the advantage of throwing your own exception is that often you can provide better diagnostics.
It is strongly recommended that you not catch general
exceptions such as Throwable, Exception,
or Error. Doing so can be potentially problematic.
2.6. Should a test call
the System.exit method?
No. Depending on how you run the tests, you may get a security exception from the harness.
2.7. Can a test write to
System.out and System.err?
Yes. The harness will capture everything sent to both streams.
2.8. Can a test check
the output of System.out and
System.err?
Yes, compiler tests using the @compile tag can use
the /ref=file option. Such tests are generally
not recommended, since the output can be sensitive to the locale in
which the are run, and may contain other details which may be hard
to maintain, such as line numbers.
While not part of the tag specification, some tests use the
string "/nodynamiccopyright" after the
@test tag to indicate that that the file should not be
subject to automated copyright processing that might affect the
operation of the test.
2.9. What should I do with a test once I've written it?
Check it into the test directory in your workspace.
Yes, it really is that simple.
2.10. Where is the test directory?
All JDK workspace beginning with JDK 1.1.5 contain a top-level
directory named test. If your workspace doesn't yet
contain this directory, do a bringover from your integration
workspace.
-
% bringover -p <workspace> test
This will bringover just the test directory from the parent workspace <workspace>.
2.11. Why not have separate test workspaces that only contain tests?
Checking a test into the workspace against which it's first written helps prevent spurious test failures. In the case of a regression test, this process ensures that a regression test will migrate upward in the workspace hierarchy along with the fix for the corresponding bug. If tests were managed separately from fixes then it would be difficult to distinguish between a true test failure and a failure due to version skew because the fix hadn't caught up with the test.
2.12. How are the test directories organized?
Tests are generally organized following the structure of the
Java API. For example, the test directory contains a
java directory that has subdirectories
lang, io, util, etc.
Each package directory contains one subdirectory for each class
in the package. Thus tests for java.io.FileInputStream
can be found in java/io/FileInputStream.
Each class directory may contain a combination of single-file tests and further subdirectories for tests that require more than one source file.
2.13. What should I do if my test directory is missing a subdirectory that I need?
The test directory in your workspace may not contain a particular subdirectory since the source code management system creates directories only if they contain files that have been put under source code control. If you run into this problem, just create the directory that you need and check your test files into it. The source code management system will take care of the rest.
2.14. How should I name a test?
In general, try to give tests names that are as specific and
descriptive as possible. If a test is checking the behavior of one
or a few methods, its name should include the names of those
methods. For example, a test written for a bug in the skip method
of FileInputStream could be placed in
test/java/io/FileInputStream/Skip.java. A test written
for a bug that involves both the skip and available methods could
be named SkipAvailable.java.
Tests that involve many methods require a little more creativity
in naming, since it would be unwieldy to include the names of all
the methods. Just choose a descriptive word or short phrase. For
example, a test that checks the general behavior of a
FileInputStream after end-of-file has been reached could be named
AfterEOF.java.
It can be helpful to add more information to the test name to
help further describe the test. For example, a test that checks the
skip method's behavior when passed a negative count could be named
SkipNegative.java.
You might find that the name you want to give your test has
already been taken. In this case either find a different name or,
if you're just not in a creative mood, append an underscore and a
digit to an existing name. Thus if there were already a
Skip.java file, a new test for the skip method could
be named Skip_1.java.
Some tests require more than one source file, or may need access
to data files. In this case it's best to create a subdirectory in
order to keep related files together. The subdirectory should be
given a descriptive mixed-case name that begins with a lowercase
letter. For example, a FileInputStream test that needs an input
file as well as its Java source file in order to test the
interaction of the read and available methods could be placed in
the subdirectory
test/java/io/FileInputStream/readAvailable.
Some tests involve more than one class in a package, in which
case a new subdirectory in the relevant package directory should be
created. For example, a set of general tests that exercise the
character streams in the java.io package could be
placed in the test/java/io/charStreams directory.
2.15. What about tests that don't fit into the API structure?
In addition to a java directory for API-related
tests, the test directory contains a javax directory,
vm directory, a tools directory, and
com and sun directories.
2.16. Can tests in different directories have the same name?
Yes. When a test is run by the harness, a special classloader is used so that the classpath is effectively set to include just the directory containing the test, plus the standard system classes. Thus name clashes between tests in different directories are not a problem.
An alternative approach would be to associate a different package with each test directory. This is done, for example, in the JCK test suite. The difficulty with this idea is that in order to debug a test (under dbx or workshop or jdb or whatever) you must set up your classpath in just the right way. This makes it difficult to diagnose bugs that are reported against specific tests.
2.17. How do I write a test for an AWT bug or a Swing bug?
Bugs in the graphical facilities of the JDK generally require
manual interaction with applets. Applet tests are written in much
the same way as the simple main tests described above.
The primary differences are that a second "@" tag is given to
indicate that the test is an applet test, and an appropriate HTML
file is needed. For example, an AWT test named
Foo.java would have the form:
-
/* @test 1.1 97/10/12 @bug 9876543 @run applet/manual Foo.html */ public class Foo extends java.awt.Applet { ... }
or
-
public class Foo extends javax.swing.JApplet { ... }
The @run tag tells the harness how to run the test.
The first argument is the run type, applet, followed
by an option, /manual, that flags this test as a
manual test requiring user interaction. The remaining arguments to
the @run tag are passed to the program in a manner
appropriate to the run type. In this case, the test will be run
just as if the appletviewer had been invoked on
Foo.html. Thus Foo.html must contain, at
least, an HTML applet tag with any necessary
parameters.
2.18. How does the user know what to do for a manual applet test?
When the harness runs a manual applet test, it will display the contents of the HTML file that defines the applet. Include instructions in the HTML file so that the person running the test can figure out what to do if any interaction is required.
2.19. Exactly what
does the /manual option mean?
The /manual option indicates to the harness that
this is a manual test. This allows the harness to distinguish
manual from automatic tests, which is important since the latter
can be run without user interaction.
There are actually three kinds of applet manual tests:
Self-contained tests, yesno tests, and
done tests.
A self-contained manual test handles all user interaction
itself. If the test fails, whether this is determined by the user
or by the applet, then the applet must throw an exception.
Self-contained tests specify applet/manual for the
first @run argument.
A yesno test requests the harness to ask the user
whether the test passes or fails. To do this, the harness will put
up pass and fail buttons, and it's up to
the user to inspect the screen and click one of the buttons. The
harness will take care of shutting down the applet. The test will
also fail if the applet throws an exception. Yesno
tests specify applet/manual=yesno for the first
@run argument.
A done test requests the harness to put up a
done button. After the user has completed whatever
actions are required by the test, the user clicks done
and the harness shuts down the applet. The program must itself
determine whether the test is to pass or fail, and throw an
exception in the latter case. Done tests specify
applet/manual=done for the first @run
argument.
main and shell may also specify the
manual option using main/manual and
shell/manual respectively. These tests must be
completely self-contained.
2.20. How does a manual applet test indicate success or failure?
Just as with main tests, an applet may throw an
exception at any time to indicate test failure. A done
test applet, for example, can check for failure in the
java.applet.Applet.destroy method. If an applet
doesn't throw an exception, and if, in the case of a
yesno test, the user clicks pass, then
the test is considered to have passed.
Be very careful about where failure is checked. The AWT event thread does not propagate exceptions!
2.21. What if the bug I'm fixing requires a more complex regression test?
As time goes on the test tag syntax will be enriched to handle various cases not considered here. If you can't write a test using the tags described here, check in a regression test anyway. If you include a tag of the form:
-
@run ignore
in the source file, then the harness will not attempt to run the test. In order to allow a human to run the test manually, describe as carefully as you can in a comment how to do so. Finally, to help guide the evolution of the test tags, see the OpenJDK page a suitable forum or mailing list to contact, mentioning the test as well as any suggestions you might have about how to extend the harness to handle it.
3. Getting Started
3.1. What does a typical
invocation of jtreg look like? How can I make sure
that I can even run the JavaTest harness?
You may verify that the JavaTest harness can be properly invoked
by using jtreg to run this sample test.
-
/* * @test * @bug 2718282 * @summary Hello test */ public class Hello { public static void main(String [] args) throws Exception { if (true) System.out.println("Hello World!"); else throw new Exception("??"); } }
A typical invocation of jtreg on that test is:
-
ribbit$ jtreg -verbose:all -testjdk:/usr/local/java/jdk1.4/solsparc Hello.java
where
-verbose:allis a verbose option which causes output from all tests (regardless of whether they passed or failed) to be printed at completion of the test.-testjdkspecifies the location of the JDK version which should be used to run the test.
Modulo the line numbers, output for the successful invocation of
jtreg will look like:
-
1 -------------------------------------------------- 2 TEST: Hello.java 3 JDK under test: (/usr/local/java/jdk1.4/solsparc/jre) 4 java version "1.4.0-beta" 5 Java(TM) 2 Runtime Environment, Standard Edition (build 1.4.0-beta-b56) 6 Java HotSpot(TM) Client VM (build 1.4-beta-B56, mixed mode) 7 8 ACTION: build -- Passed. Compilation successful 9 REASON: Named class compiled on demand 10 TIME: 3.024 seconds 11 12 ACTION: compile -- Passed. Compilation successful 13 REASON: .class file out of date or does not exist 14 TIME: 3.023 seconds 15 STDOUT: 16 STDERR: 17 18 ACTION: main -- Passed. Execution successful 19 REASON: Assumed action based on file name: run main Hello 20 TIME: 0.862 seconds 21 STDOUT: 22 Hello World! 23 STDERR: 24 STATUS:Passed. 25 26 TEST RESULT: Passed. Execution successful 27 -------------------------------------------------- 28 test results: passed: 1 29 Report written to /u/iag/jtwJTreport/report.html 30 Results written to /u/iag/jtw/JTwork
The test was compiled and executed. No exception was thrown during execution, thus the test passed.
Interpretation of this output is as follows:
- line 2 - The name of the test that was run.
- line 3 - The JDK under test (should be identical to the value
passed via the
-testjdkoption). - line 4-6 - The product version produced when
java [-JVMOptions]version" is called for the JDK under test. Valid[-JVMOptions]include-client,-server,-hotspot,-d64, and-d32, as applicable to the current platform and test JDK. - lines 8-10, 12-16, 18-24 - The set of actions that were run according to the test description provided. Each action contains five parts.
-
- the name of the action and its final status
- the reason the action was taken
- the amount of time to run the test
- standard output (note line 22 of the
mainaction contains the string "Hello World!") - standard error
- line 26 - The final result of the test.
- lines 28-30 - Summary information about all the tests that were run.
-
- line 28 - One test passed. This line would also indicate the number of tests that failed, or that produced errors, as applicable.
- line 29 - Location for
.htmlreports. - line 30 - Location for auxiliary files generated during the
test execution. Of particular note are the results files
(
.jtr) which contain information about the individual tests that were run.
3.2. Bleah! That verbose output is so long! Can I have something shorter?
Yes. Several different options provided with jtreg
influence the output per test. Here are a few verbose settings in
order of decreasing average output per test.
-verbose:fail(and related-verbose:pass,-verbose:error, and-verbose:all)-verbose-verbose:summary- no verbose option
The following samples of output correspond to each of the above
settings. Each sample is run with three tests:
Pass.java, Fail.java, and
Error.java . Note that in some cases, the output
varies substantially depending on whether the test passed or
failed.
-verbose:fail -
Full output for failed tests only. Two lines for tests that passed
or produced errors (related options: -verbose:pass,
-verbose:fail, and -verbose:all).
-
ribbit$ jtreg -verbose:fail Pass.java Fail.java Error.java -------------------------------------------------- TEST: Pass.java TEST RESULT: Passed. Execution successful -------------------------------------------------- TEST: Fail.java JDK under test: (/usr/local/java/jdk1.4/solsparc) java version "1.4.0-beta" Java(TM) 2 Runtime Environment, Standard Edition (build 1.4.0-beta-b56) Java HotSpot(TM) Client VM (build 1.4-beta-B56, mixed mode) ACTION: build -- Passed. Compilation successful REASON: Named class compiled on demand TIME: 3.649 seconds ACTION: compile -- Passed. Compilation successful REASON: .class file out of date or does not exist TIME: 3.637 seconds STDOUT: STDERR: ACTION: main -- Failed. Execution failed: `main' threw exception: java.lang.Exception: Fail REASON: Assumed action based on file name: run main Fail TIME: 1.219 seconds STDOUT: STDERR: java.lang.Exception: I failed at Fail.main(Fail.java:5) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:30) at sun.reflect.InflatableMethodAccessorImpl.invoke(InflatableMethodAccessorImpl.java:46) at java.lang.reflect.Method.invoke(Method.java:306) at com.sun.javatest.regtest.MainWrapper$MainThread.run(MainWrapper.java:94) at java.lang.Thread.run(Thread.java:579) JavaTest Message: Test threw exception: java.lang.Exception: I failed JavaTest Message: shutting down test STATUS:Failed.`main' threw exception: java.lang.Exception: I failed TEST RESULT: Failed. Execution failed: `main' threw exception: java.lang.Exception: I failed -------------------------------------------------- TEST: Error.java TEST RESULT: Error. Parse Exception: Unexpected length for bugid: 31415926, -------------------------------------------------- test results: passed: 1; failed: 1; error: 1 Report written to /u/iag/jtw/JTreport/report.html Results written to /u/iag/jtw/JTwork Error: some tests failed or other problems occurred
-verbose - This
option produces three lines of output per test: start, end, final
status.
-
ribbit$ jtreg -verbose Pass.java Fail.java Error.java runner starting test: Pass.java runner finished test: Pass.java Passed. Execution successful runner starting test: Fail.java runner finished test: Fail.java Failed. Execution failed: `main' threw exception: java.lang.Exception: I failed runner starting test: Error.java runner finished test: Error.java Error. Parse Exception: Unexpected length for bugid: 31415926, test results: passed: 1; failed: 1; error: 1 Report written to /u/iag/jtw/JTreport/report.html Results written to /u/iag/jtw/JTwork Error: some tests failed or other problems occurred
-verbose:summary
- A single line of output per test: final status and name of
file.
-
ribbit$ jtreg -verbose:summary Pass.java Fail.java Error.java Passed: Pass.java FAILED: Fail.java Error: Error.java test results: passed: 1; failed: 1; error: 1 Report written to /u/iag/jtw/JTreport/report.html Results written to /u/iag/jtw/JTwork Error: some tests failed or other problems occurred
No verbose option provides only general summary information about all the tests run.
-
ribbit$ jtreg Pass.java Fail.java Error.java test results: passed: 1; failed: 1; error: 1 Report written to /u/iag/jtw/JTreport/report.html Results written to /u/iag/jtw/JTwork Error: some tests failed or other problems occurred
If there is information that you find lacking in all of these options, please contact the developer to determine if it is possible to make it available.
3.3. Are there abbreviations for these long options?
Yes. The -help option to jtreg lists
the long and abbreviated forms of all options.
3.4. How do I view what
a test sends to System.out/System.err?
You have several alternatives.
- Use the
-verbose:alloption (or its result-sensitive related options-verbose:pass,-verbose:fail,-verbose:error) tojtreg. - Use the JavaTest harness GUI.
- View the test's
.jtrfile.
3.5. What is a
.jtr file?
As each test is run, it produces a JavaTest Results
(.jtr) file which contains information about how the
test was run, the name of the test, standard output, standard
input, final status, etc. The name of the file is the basename of
the file containing the test description followed by the
.jtr extension. For example, Hello.java
produces a file called Hello.jtr. These files reside
in the work directory which contains a directory hierarchy that
parallels the source structure.
3.6. What are the report and work directories?
The report directory contains all final reports in HTML format.
By default, the name of the report directory is
JTreport. The name may be selected via the
-reportDir option to jtreg.
The work directory contains any files which are generated during
a test run. These include .class and .jtr
files. By default, the name of the work directory is
JTwork. The name may be selected via the
-workDir option to jtreg.
3.7. What's the difference between the "fail" and "error" return status?
If a test failed, then the harness was able to actually run the test code. The failure could either indicate that the test truly failed (i.e. it threw an exception) or that there was some problem running the test due to security restrictions, lack of resources, etc.
If error is the final result of a test, then the harness was unable to run the test. An error is most commonly associated with problems in the test description (typos, missing required arguments, etc.).
In either case the result contains a short message intended to provide hints as to where the problem lies.
3.8. If a test fails, I'd like to put all of my debugging information into the final result. How do I do that?
The final result string is composed by the harness. For tests
that fail because an exception is thrown, the result string will
contain some header string such as `main' threw
exception: followed by the exception's type and detail
message. This detail message should contain sufficient information
to provide the test user a starting point to investigate the
unexpected failure. It should not contain full debugging
information.
The harness makes no guarantees as to the availability of any detail message longer than 128 characters.
3.9. I've heard that the
jtreg has a GUI. How do I access that?
The complete JavaTest harness GUI is available via the
-gui option to jtreg. The Online Help,
available on the Help menu, provides a detailed
description of the graphical user interface. You can also use the
F1 key to get context sensitive help about any
selected item in the graphical user interface.
3.10. Can I verify the correctness of test descriptions without actually running the tests?
Yes! The -check option to jtreg will
find test descriptions and will determine whether they are written
according to the tag specification. Tests will not be
executed.
The following sample output illustrates use of this option.
-
ribbit$ jtreg -verbose:summary -check Pass.java Fail.java Error.java Passed: Pass.java Passed: Fail.java Error: Error.java test results: passed: 2; error: 1 Report written to /u/iag/jtw/JTreport/report.html Results written to /u/iag/jtw/JTwork Error: some tests failed or other problems occurred
3.11. Can I generate reports for tests that have already been run?
Yes! The -reportOnly option to jtreg
will generate the standard HTML reports for tests which have been
previously executed. Tests will not be executed. A
work directory containing the results of the
executed tests must be provided. The default location is
./JTwork. An alternate directory may be specified
using -workDir.
3.12. How do I run
jtreg under Windows?
jtreg is normally invoked by a wrapper script,
written for the Bourne family of shells. On Windows, you can use
MKS, which uses ksh, or
you can use Cygwin, which uses
ash or bash, depending on which version you are using.
You can also start jtreg directly, with a command
of the form java -jar jtreg.jar options, but
you will still need MKS or Cygwin installed to be able to run shell
tests.
3.13. Which should I use? MKS or Cygwin?
jtreg supports both, equally. However, the tests in
the JDK regression test suite assume that MKS is available. So,
when you are writing tests for that test suite, you should make
sure that your test at least works with MKS. If you prefer to use
Cygwin, and can make your test run with both, so much the better.
3.14. How do I run only tests which were written for a specific bugid?
The -bug option to jtreg will run only
those tests which define the given bugid using the
@bug tag.
3.15. How do I run only tests NOT requiring manual intervention?
The -automatic option to jtreg will
ignore all tests which contain the /manual tag
option.
3.16. How do I run only tests requiring manual intervention?
The -manual option to jtreg will run
only those tests which contain the /manual tag
option.
3.17. How do I specify VM options to the test's JVM?
Most JVM options is recognized by jtreg and passed
to the test. Note that these options are sentitive to both the
platform and the JDK being tested. For example, Linux does not
currently support 64-bit operation. Thus, the option
-d64 is not valid on Linux and will be rejected.
4. General Problems
4.1. My test only passes if I don't use jtreg to run it. Why does it fail in jtreg?
By default, tests run using jtreg are each run in a
separate JVM. By design, only a limited number of shell environment
variables are passed down to the test's JVM. This may affect how a
test is run.
As per spec, the only shell environment variables that are propagated into the test's JVM are:
- Linux and Solaris:
PATHis set to/bin:/usr/bin- The following are propogated from the user's environment:
DISPLAY,HOMELANG,LC_ALL,LC_TYPE,LPDEST,PRINTER,TZandXMODIFIERS
- Windows:
PATHis set to the MKS or Cygwin toolkit binary directory- The following are propogated from the user's environment:
SystemDrive,SystemRootwindir
A test which relies on any other environment variables will experience difficulties running in jtreg.
4.2. How do I set the
CLASSPATH environment variable for my test?
The harness sets the CLASSPATH for the
compile, main, and applet
actions to be the system class path plus the test's source and
classes directory.
It is possible to set the classpath for the
main/othervm action via the -classpath
option to java. Any other modification of the
CLASSPATH must be done using the shell
action.
4.3. Why don't you just pass all of my shell environment variables to the JVM running the test?
The approach of passing down a list of pre-defined environment variables helps guarantee consistent results across different people running the test(s).
4.4. Why is the default to run tests in another JVM?
Insulation from other tests. Most well-behaved tests do not modify the JVM; however, if a test does modify the JVM it is possible that this change will interfere with subsequent tests. Running each test in another JVM allows for the possibility of bad test suite citizens.
4.5. Why would I ever want to run in the same JVM?
Speed.
4.6. Should a test call
the System.exit method?
NO! The default harness security manager prevents tests
from calling System.exit. If the test is running in
the same JVM as the harness and the harness was not permitted to
install its own security manager, a call to
System.exit will cause the harness itself to exit!
If the test is running in its own separate JVM, a call to
System.exit may not allow the harness to properly
handle test termination.
4.7. My test only applies to one platform and it will fail/not run in others. How do I prevent the harness from running it on the wrong platform?
The tag specification provides no way
to indicate any platform requirements. If the test only applies to
a single platform, then the test itself must determine the current
platform and decide whether the test should be run there. If the
test suite is running on the wrong platform, the test should pass
(i.e. just return) otherwise, the test should proceed. A
significant benefit to this approach is that the same number of
tests in a testsuite will always be run if the same arguments are
given to jtreg regardless of the particular
platform.
For tests that are written in Java code (i.e.
applet and main tests), you may determine
the platform via the system properties. The following code fragment
may be used to distinguish between SunOS sparc, SunOS x86, Windows,
etc.
-
String arch = System.getProperty("os.arch"); String name = System.getProperty("os.name"); if (arch.equals("sparc")) { System.out.println("SunOS, sparc"); } else if (arch.equals("x86")) { if (name.equals("SunOS")) System.out.println("SunOS, x86"); else if (name.startsWith("Windows")) System.out.println("windows"); else throw new RuntimeException("unrecognized OS:" + " os.arch == " + arch + " os.name == " + name); } else if (arch.equals("i386")) { System.out.println("Linux, i386"); } else { throw new RuntimeException("unrecognized system:" + " os.arch == " + arch); }
This approach is not suitable for shell tests. In
this case, you can determine the platform via uname.
The following code accomplishes the same task as above.
-
OS=`uname -s` case "$OS" in Linux ) echo "Linux" ;; SunOS ) HARDWARE=`uname -m` case "$HARDWARE" in sun4u ) echo "SunOS, sparc" ;; i86pc ) echo "SunOS, x86" ;; * ) echo "unrecognized SunOS: $HARDWARE" ;; esac ;; Windows_* ) echo "windows" ;; * ) echo "unrecognized system: $OS" ;; esac exit $?
4.8. How can I make
applet and main action tests read from
data files?
When jtreg is executed, it cd's into a scratch area
to ensure that a test can not alter the test suite. Thus, a direct
reference to a data file without some sort of indicator as to where
the test was originally located will fail.
The system property test.src contains the name of
the directory where the test source resides. The following example
illustrates how to read the data file foo which is
contained in the test source directory. Note that the default value
of "." allows this test to run both with the harness,
and without (in the source directory).
-
File f = new File(System.getProperty("test.src", "."), "foo"); InputStream in = new File(f);
4.9. Can I use
package statements in my tests?
Yes…but you probably don't want to. The harness searches
for class files in a class directory with components which are
parallel to the source directory. It will be unable to locate
packaged class files when the test is invoked via reflection. Use
of the package statement is not recommended unless the
test is intended to test package statements.
Tests which test the package mechanism may use package statements; however, it will be the responsibility of the test writer to properly execute the compiled class as necessary.
4.10. Why can't multiple test source files in the same directory have package-private classes of the same name?
In the Java language, package private classes defined in different files in the same directory are interpreted as duplicate class definitions. The contents of the class files depends on the order of compilation. To avoid compilation order dependency problems, we recommend that you define auxiliary classes as inner classes.
For performance reasons, the harness does not automatically remove class files between individual tests or build each test within its own unique subdirectory. This allows us to cache class files across test runs and share code between tests in the same directory.
4.11. Should a test
catch Throwable, Exception, or
Error?
Ideally, only specific, anticipated exceptions should be caught by a test. Any other exception which is provoked during testing should be propagated back to the harness.
In the event that a very general exception is caught in test
code, a certain amount of care will need to be exercised. For
example if a user wants to stop the harness during a test run, an
InterruptedException is used to interrupt the test. If
that exception is caught by a test and not re-thrown to the
harness, the stop request will be lost and the tests will not
stop!
Here is a list of exceptions that may need to be re-thrown:
InterruptedException(fromException)InterruptedIOException(fromIOException)ThreadDeath(fromError)
4.12. My test requires
that I use information printed to System.out or
System.err to determine whether a test passed or
failed. When I run my test in the harness, I can't seem to find
these output streams.
Currently, information sent to System.out or
System.err is only available after the test
has finished running.
Note that this question indicates that the test itself can not
determine whether it passed or failed (i.e. it needs human
intervention). Thus, the test uses the manual option.
The suggestions provided for the applet
action may apply.
4.13. My test does
tricky things that are not supported by jtreg. Can I
still write a regression test?
Yes. Most tests can be written using a series of
main, clean, build,
applet, and compile actions. However,
there have been a few tests that need to do things like run a
specific application or access specific environment variables. The
shell action allows a user to invoke a Bourne
shell-script which can run arbitrary commands, including running
java and javac.
Warning! All tests, including shell-script tests, may be run on multiple platforms including Linux, Solaris, Windows NT, and Windows 95/98. The shell-script should be written to with this in mind. The following code fragment may be useful in defining various platform-dependent variables.
-
OS=`uname -s` case "$OS" in SunOS | Linux ) NULL=/dev/null PATHSEP=":" FILESEP="/" TMP=/tmp ;; Windows_* ) NULL=NUL PATHSEP=";" FILESEP="\\" TMP=$TEMP ;; * ) echo "Unrecognized system!" exit 1; ;; esac
If the shell action still does not provide the
flexibility needed to write the regression test, then use the
ignore action. It is also advisable to include a
comment with sufficient detail to allow a person to run the test
and verify its behavior.
4.14. What happens if my test returns when there are still threads running?
The harness runs the main action's
main method in its own thread group. The thread group
will be destroyed by the harness when the main method
returns. It is the responsibility of the test to return only after
the appropriate tasks have been completed by its subsidiary
threads.
4.15. If my bug hasn't been fixed, and the test is run, the JVM crashes. How do I make sure that the test doesn't cause the harness to crash?
If the symptom of a bug is a JVM crash, then the test's
description should include the othervm option. This
will allow the harness to continue running any subsequent tests,
write its reports, and exit normally.
4.16. The JavaTest harness is running into problems running the test because of issues with the JDK I'm trying to test. What can I do?
When the harness is used to run tests, two possibly different versions of the JDK are used: the JDK version used to run the harness and the JDK version used to run the test(s) themselves.
To run the harness with one version of the JDK and the tests
with another, use the -othervm option in conjunction
with the -testjdk option. The -testjdk
option will specify the version of the JDK to run the tests. The
environment variables JT_JAVA or
JAVA_HOME will specify the version of the JDK for the
harness.
4.17. My test requires that I install my own security manager, but it appears that the JavaTest harness has already installed one. What do I do?
The harness normally installs its own rather permissive security manager in self-defense to prevent tests from interfering with each other. The harness' security manager is designed to prevent changes to the JVM's state that would impact other tests. Most tests will not find the standard harness security manager a hindrance.
A test which must install its own security manager will always
need to run in its own separate JVM. To do this, add the
othervm option to the main action in the
test description.
4.18. Can I automate running regtests or can I run the tests on a regular basis?
Yes. If you are using a UNIX system, man crontab is
your friend. Other platforms have similar facilities (often third
party) for automated command scheduling.
4.19. I run all (or a
huge part) of the regression test suite as part of a cron job or
other nightly process. I'd like to generate my own reports or I'd
like to send myself e-mail whenever a test fails. Do I have to
parse the verbose output or the .jtr file?
No. The harness supports an observer interface. APIs exist to query test results at the conclusion of the test's execution. A user can write their own observer to record and act on information as desired.
5. Tag Problems
5.1. How do I decide
whether my test should use the compile action or the
build action?
The build action will compile the specified class
only if the classfile doesn't exist or is older than its source.
The compile action always invokes the
compiler. Typically, the compile action is used only
to test the compiler, while the build action is used
for tests that make use of multiple sources or for API tests.
5.2. When do I need to
specify the build action?
Each main and applet action contains
an implied build action. The harness will build the
class specified by the main or applet
actions as needed without any prompting. If the test requires
additional class(es), every additional class must be associated
with an explicit build action.
5.3. How do I decide
whether my applet test should use the main action or
the applet action?
Ultimately, that decision is left to the person writing the test; however, the following should be considered.
Tests which use the applet action are not
necessarily restricted to tests which must run in a browser. Any
Swing/AWT code which can be written such that it derives from
java.applet.Applet or javax.swing.JApplet
is a potential applet test.
For tests which test graphics functionality, there are three
major advantages to selecting the applet action over
the main action: expanded manual support
leading to less duplicated code per test, thread synchronization,
and cleanup.
Frequently, tests which test graphics functionality need some
sort of user interaction to determine whether the test behaves as
expected. The applet action takes care of providing a
user interface which contains instructions for the user and the
appropriate interface to indicate pass,
fail, or done as indicated by the
manual option. User instructions are taken from the
.html file referenced in the applet
action. Each main action which tests graphics
functionality must implement their own version of this interface.
This path leads to more code needed per test and less consistency
across tests in the test suite.
A main action test is deemed to be completed when
the main method returns. A test which requires
multiple threads must take care not to allow the main method to
return before those other threads have completed. The
applet action handles basic AWT thread
synchronization.
Finally, the applet action handles test cleanup. If
a test can not or does not dispose top-level windows or any AWT
threads, they will be eliminated by the harness after the test
completes.
5.4. I put in an
ignore tag into my test description but my test wasn't
ignored.
The ignore tag should be used for tests that are
too complex for the currently defined set of tags or for tests that
should be temporarily ignored. The ignore tag
instructs the harness to ignore that and any subsequent
tags. Check the location of the ignore tag.
5.5. Can I use the
@author, @run, etc. tags in other
files?
Yes. The tags may be used for documentation purposes in any
file. Only those comments whose leading tag is @test
is considered a test description.
6. Applet Problems
6.1. My
/manual test sends events to
System.out/System.err so that the user can determine
whether the test behaved properly. How do I write my test if I
can't see these output streams?
The test code should be written to determine whether a test has
passed or failed based on events generated during a given
time-frame. Use the /manual=done option of the
applet action to set the time frame. If the user has
not generated the expected event before the done
button has been pressed, the test should detect this in the
destroy method and throw an exception.
While this approach takes potentially more time to implement, it avoids user error which may occur in checking the event. This scheme also avoids string comparison of events. (A much safer way to determine whether the expected event has been received is to check the event type, coordinates, modifiers, etc.)
Warning! The AWT event thread does not propagate
exceptions! It is recommended that all exceptions indicating
failure of the test be thrown from one of the methods called by the
harness. (i.e. init(), start(),
stop(), destroy())
The following simple applet test illustrates the
recommended behavior.
Basic .html test description file.
-
<html> <body> <!-- @test @bug 2997924 @summary Sample test that verifies an event @run applet/manual=done SampleTest.html --> <applet code=SampleTest width=200 height=50></applet> Select the "pick me" check box. </body> </html>
The sample test code.
-
import java.applet.Applet; import java.awt.Checkbox; import java.awt.FlowLayout; import java.awt.Panel; import java.awt.event.ItemEvent; import java.awt.event.ItemListener; // WARNING! The AWT event thread does not propagate exceptions! // It is recommended that all exceptions indicating failure // of the test be thrown from one of the methods called by the harness. // (i.e. init(), start(), stop(), destroy()) public class SampleTest extends Applet { public void init() { setLayout(new FlowLayout()); add(new TestPanel(this)); } public void destroy() { if (myEvent == null) throw new RuntimeException("no events"); else { Checkbox cb = (Checkbox)(myEvent.getItemSelectable()); if (!(cb.getLabel().equals("pick me!") && cb.getState())) throw new RuntimeException("unexpected last event"); } } class TestPanel extends Panel { Checkbox pickMe, notMe; Listener listener = new Listener(); Applet applet; public TestPanel(Applet myApplet) { applet = myApplet; pickMe = new Checkbox("pick me!"); notMe = new Checkbox("not me"); pickMe.addItemListener(listener); notMe.addItemListener(listener); add(pickMe); add(notMe); } class Listener implements ItemListener { // Don't throw an exception here. The awt event thread // will NOT propagate your exception! public void itemStateChanged(ItemEvent event) { System.out.println("event: " + event); myEvent = event; } } } private ItemEvent myEvent; }
6.2. I threw an
exception, the output was sent to System.err, but my
test still passed. What happened?
Verify that the exception was not thrown by the event thread.
The event thread does not propagate exceptions. Furthermore, the
event thread is in a separate thread group and the harness cannot
catch exceptions thrown from there. It is strongly
recommended that all exceptions indicating failure of the test be
thrown from one of the methods called by the harness. (i.e.
init(), start(), stop(),
destroy())
6.3. My
applet action test didn't run my main
method!
applet action tests do not call main.
A test which uses the applet action is run by invoking
its init, start, and
setVisible(true) methods. Depending on the value of
manual, the harness will pause either for a few
seconds or until the user clicks on the pass,
fail, or done buttons. Finally, the
harness will invoke the stop and destroy
methods.
The main method of an applet action
will only be used if the test was run outside of the harness.
6.4. If I have an applet
test, do I put the test description in the .html file
or the .java file?
It doesn't matter. When jtreg is run on a test
suite or directory, the test description will be found regardless
of the file particulars. When running a single test,
jtreg must be invoked on the file which contains the
test description.
6.5. For my
/manual tests, how do I provide the user instructions
to run the test?
User instructions should be provided in the applet's HTML file.
The uninterpreted HTML file will be displayed by the
applet action in a TextArea labelled html file
instructions:.
6.6. For
/manual tests, how is the initial size of the running
applet determined?
The size of the applet is statically determined by the
height and width attributes provided to
the HTML applet tag. The applet interface provides a
way to dynamically change the size of the applet by setting the
applet size: to "variable".
7. Deciphering Common Harness Errors
7.1. Failed.
Unexpected exit from test
Answer: The test has completed in an unexpected manner.
This could be caused by some sort of fault (e.g. a segmentation
fault) or because the harness has detected a call to
System.exit from the test.
Tests are not allowed to call System.exit because
the test must have the ability to run in the same JVM as the
harness. Calling System.exit while the test is running
in this manner whould cause the harness itself to exit! Instead of
calling System.exit(), throw an exception.
Be warned that the AWT event thread does not propagate
exceptions, so if the test was exiting from the event thread, it is
not sufficient to simply throw an exception. The test must set some
variable which can be used to throw an exception from one of the
methods called by the harness. (i.e. init(),
start(), stop(), or
destroy())
7.2. Error. Can't
find `main' method
More symptoms: In System.err, you get a
stack trace for an java.lang.NoSuchMethodException and
some harness messages.
-
java.lang.NoSuchMethodException at java.lang.Class.getMethod(Class.java) at com.sun.javatest.regtest.MainWrapper$MainThread.run(MainWrapper.java:89) at java.lang.Thread.run(Thread.java) JavaTest Message: main() method must be in a public class named JavaTest Message: ClassNotPublic in file ClassNotPublic.java
Answer: The class defining the test must be declared
public and the class name must be the basename of the
.java file; otherwise the harness will not be able to
use reflection to invoke the test.
7.3. Error. Parse
Exception: No class provided for `main'
Answer: An @run main tag was provided
without the required class name. Either provide the name of the
class or remove the line entirely if appropriate.
The line may be removed without impacting the test if all of the following criteria are met:
- The file containing the test description has the
.javaextension. - This is the only
@runtag in the test description. - No options to
mainare required
In removing the line, we take advantage of the default action
for .java files.
7.4. Error. Parse
Exception: `applet' requires exactly one file
argument
Answer: The applet action requires a single argument
which should be the name of the .html file which
contains (at minimum) the HTML applet tag.
7.5. Error. Parse
Exception: `archive' not supported in file: …
More Symptoms: The test is an applet action
test. The HTML applet tag includes the
archive attribute.
Answer: The regression extensions to the harness do not
support the archive attribute.
7.6. test
results:no tests selected
More Symptoms: At a terminal window, you get:
-
test results:no tests selected Report written to /home/iag/work/doc/JTreport/report.html Results written to /home/iag/work/doc/JTwork
Answer: No test descriptions were found by
jtreg in the file or directory specified. If the
-automatic option to jtreg was provided,
then there were no tests which did not include the
/manual tag option.
Verify that the first tag of each test description is
@test.
7.7. Test does not
have unique name within work directory
More Symptoms: At a terminal window, you get:
-
Error: -- Test does not have unique name within work directory -- Test name: Duplicate -- Used by test: Duplicate.java -- And by test: Duplicate.html -- Overwriting results of first test
A single directory contains more than one file with the same basename.
Answer: The two files contain descriptions of tests and
the harness is unable to determine a unique .jtr filename so the harness will overwrite
the results of the first test. It is possible to have a single
directory with more than one file with the same basename; however,
only one of those files may have a test description
(@test is the first token of the comment).
If both files contain identical test descriptions, select one
file to contain the primary test description. Disable the other
test description by either removal of the entire comment or simply
the @test tag.
If the files contain unique test descriptions, one of the basefile names must be changed.

