Example 9-8. Sample test fixture
package chap9;
import java.io.*;
import java.net.*;
import java.util.*;
// JAXP used for XSLT transformations
import javax.xml.transform.*;
import javax.xml.transform.stream.*;
// JDOM used for XML parsing and validation
import org.jdom.*;
import org.jdom.input.*;
// JUnit classes
import junit.framework.Test;
import junit.framework.TestCase;
import junit.framework.TestSuite;
import junit.textui.TestRunner;
/**
* An example JUnit test. This class performs an XSLT transformation
* and validates the result.
*/
public class SampleUnitTest extends TestCase {
private String workingDir;
// input XML files
private File aidanXMLFile;
private File johnXMLFile;
// a stylesheet that condenses the XML data
private File condenseXSLTFile;
// the transformation results
private File aidanCondensedXMLFile;
private File johnCondensedXMLFile;
private TransformerFactory transFact;
/**
* All JUnit tests have a constructor that takes the test name.
*/
public SampleUnitTest(String name) {
super(name);
}
/**
* Initialization before each test[...] method is called.
*/
public void setUp( ) {
// locate a file named test.properties in the chap9 package
ResourceBundle rb = ResourceBundle.getBundle("chap9.test");
this.workingDir = rb.getString("chap9.workingDir");
assertNotNull(workingDir);
assert("Unable to locate " + this.workingDir,
new File(this.workingDir).exists( ));
this.aidanXMLFile = new File(workingDir + File.separator
+ "aidan.xml");
this.johnXMLFile = new File(workingDir + File.separator
+ "john.xml");
this.condenseXSLTFile = new File(workingDir + File.separator
+ "condensePerson.xslt");
this.aidanCondensedXMLFile = new File(this.workingDir + File.separator
+ "aidanCondensed.xml");
this.johnCondensedXMLFile = new File(this.workingDir + File.separator
+ "johnCondensed.xml");
this.transFact = TransformerFactory.newInstance( );
}
/**
* Clean up after each test[...] method
*/
public void tearDown( ) {
// the transformation results could be deleted here, but the
// cleanup code is intentionally commented out so the
// developer can see the generated files:
// this.aidanCondensedXMLFile.delete( );
// this.johnCondensedXMLFile.delete( );
}
/**
* An individual unit test.
*/
public void testTransformWithTemplates( ) throws Exception {
Templates templates = this.transFact.newTemplates(
new StreamSource(this.condenseXSLTFile));
Transformer trans = templates.newTransformer( );
// do two transformations using the same Transformer
trans.transform(new StreamSource(this.aidanXMLFile),
new StreamResult(this.aidanCondensedXMLFile));
trans.transform(new StreamSource(this.johnXMLFile),
new StreamResult(this.johnCondensedXMLFile));
// validate both files
validateCondensedFile(this.aidanCondensedXMLFile,
"Aidan Garrett Burke", "6/25/1999");
validateCondensedFile(this.johnCondensedXMLFile,
"John Fitzgerald Kennedy", "5/29/1917");
}
/**
* Another unit test.
*/
public void testTransformer( ) throws Exception {
Transformer trans = this.transFact.newTransformer(
new StreamSource(this.condenseXSLTFile));
trans.transform(new StreamSource(this.aidanXMLFile),
new StreamResult(this.aidanCondensedXMLFile));
validateCondensedFile(this.aidanCondensedXMLFile,
"Aidan Garrett Burke", "6/25/1999");
}
// a helper method used by each of the unit tests
private void validateCondensedFile(File file, String expectedName,
String expectedBirthDate) {
try {
// first do a simple validation against the DTD
SAXBuilder builder = new SAXBuilder(true); // validate
Document doc = builder.build(file);
// now perform some additional checks
Element nameElem = doc.getRootElement( ).getChild("name");
assertEquals("Name was not correct",
expectedName, nameElem.getText( ));
Element birthDateElem = doc.getRootElement( ).getChild("birthDate");
assertEquals("Birth date was not correct",
expectedBirthDate, birthDateElem.getText( ));
} catch (JDOMException jde) {
fail("XML was not valid: " + jde.getMessage( ));
}
}
/**
* @return a TestSuite, which is a composite of Test objects.
*/
public static Test suite( ) {
// uses reflection to locate each method named test[...]
return new TestSuite(SampleUnitTest.class);
}
/**
* Allow the unit tests to be invoked from the command line
* in text-only mode.
*/
public static void main(String[] args) {
TestRunner.run(suite( ));
}
}
First, notice that SampleUnitTest extends from
junit.framework.TestCase. Each subclass of
TestCase defines a fixture and can contain
multiple individual unit tests. Each method that begins with the word
"test" is a unit test. All of the private fields in
SampleUnitTest are specific to our particular
needs and are not part of the JUnit framework.
The constructor takes the name of a unit test as an argument:
public SampleUnitTest(String name) {
super(name);
}
The name argument is the test method name, and
JUnit uses the Java reflection API to locate and instantiate the
correct method. As we will see in a moment, this constructor is
rarely called directly.
The setUp( ) method is called before each unit
test is executed. As expected, this method is used to set up
preconditions before a test is executed. Its counterpart is the
tearDown( ) method, which is called just after
each test is executed. If a fixture contains four unit test methods,
then setUp( ) and tearDown( )
will each be called four times.
For our purposes, the setUp( ) method locates all
of the files that will be used for XSLT transformations. These
include XML input files, the XSLT stylesheet, and the XSLT result
targets. It also performs some simple testing:
assertNotNull(workingDir);
assert("Unable to locate " + this.workingDir,
new File(this.workingDir).exists( ));
This first unit test in our example is the
testTransformWithTemplates( ) method. Because this
method name begins with "test," JUnit can use reflection
to locate it. The job of this test is to merely perform an XSLT
transformation using JAXP's Templates
interface, delegating to the validateCondensedFile(
) method to do the actual testing. This approach is taken
because the same testing code can be shared among a group of
individual unit tests.
The validateCondensedFile( ) method performs two
levels of testing. First, the result of the transformation is
validated against its DTD. If an exception is thrown, the test fails:
fail("XML was not valid: " + jde.getMessage( ));
JUnit will intercept this failure and display the error message to
the programmer running the test. If the validation succeeds, the unit
test then uses the assertEquals( ) method to test
some of the actual XML content:
assertEquals("Name was not correct",
expectedName, nameElem.getText( ));
In this method, if the second two arguments are not equal, the
provided error message is displayed and the test fails.
One key additional method is suite( ):
public static Test suite( ) {
// uses reflection to locate each method named test[...]
return new TestSuite(SampleUnitTest.class);
}
This is useful because it automatically locates all methods whose
names begin with "test" and adds them to a test suite.
Both TestCase and TestSuite
implement the Test interface;
TestSuite is a composite of many individual
Test objects. By organizing tests into suites,
entire families of tests can be executed by running the suite. As
expected with a composite pattern, test suites can also consist of
other test suites. At some point, one top-level test suite can
directly or indirectly include every other test in the application.
Therefore, all tests can be executed with a single command.