A strong principle of eXtreme Programming (XP) is the continuous integration aspect (see the Continuous Integration article by Martin Fowler). The traditional approach has been to developing the code first, then test it and then integrate it with other applications. The continuous integration principle is to develop code, test and integrate at the same time, i.e. at any point in time, you have a functioning code along with tests and integrated.
In order to be able to do continuous integration, you need to be able to automatically run the build for your application, including passing the unit tests (based on JUnit, Cactus, HttpUnit or others). Ant is the perfect tool for this task.
The benefits of using Ant for running unit tests are as follows:
Benefits |
---|
Ant is written in Java and thus the same scripts can be used on several systems (Windows and Unix for example). Is is therefore very well suited for building java applications. |
Ant provides enough built-in and optional tasks to be able to achieve almost any needed build task without needing to use system dependent scripts. |
By being able to very quickly rerun unit test you can verify that your tests still pass when you modify some part of your code ( regression testing) |
As it is automated, it fits well in the continuous integration principle and your tests can thus be ran automatically and continously |
The sample Ant build file described below is taken from the Cactus Sample for Servlet API 2.2 project.
Create the following directories:
Directory name | Content of the directory |
---|---|
build
|
The Ant build file and some other ancillary files (properties file, ...) |
conf
|
Configuration files for the project. These are the configuration
files that will not be put into the runtime jar. They
are configuration file that need to be visible and modifyable
by a user of the project. If they were bundled in the jar they
would be hard to modify. On the other hand, we don't want a
proliferation of configuration files, so all technical
configuration files (messages to display for errors, ...)
should be put in the src directory in correct java
packages.
|
conf/test
|
Configuration files needed for testing the project only. |
docs
|
Project documentation. |
docs/skins
|
Documentation skin for Stylebook (if using Stylebook). |
docs/xdocs
|
Documentation files (XML files) for Stylebook (if using Stylebook). |
src
|
The project sources: java files + java test files + properties files + test properties files + other files for runtime or test (XML files, ...). Note that all test files should begin by 'Test' or 'test' in order to easily separate them from other files so that in an Ant target we'll be able to include only the runtime files. |
web
|
The project web files: HTML, JSP, ... (if any) |
out
directory will be created by the Ant build. All
build-generated files will be put in that directory (compiled classes,
generated javadoc documentation, test configuration files for running
an application server, ...).
lib
directory because it is always
better not to include dependent jars in your project whenever
possible for the following reasons: better continuous integration
with other libraries (meaning they also evolve and you should test
as much as possible with the latest version to discover potential
problems early, more lightweight downloads, less jar proliferation
(you'll end up with tens of the same jars otherwise), more version
control and integration checks (if your project uses 2 external
libraries that need another third library but not in the same version
you are in trouble !), ...
Define the following targets in your build.xml
.
Target name | Description | Type |
---|---|---|
init |
Defines token filters, timestamp, display some information on screen, ... | Internal |
usage |
Display usage information about the targets. | External |
prepare |
Set up the output directory where build generated files will be
put and copy the java sources to this output directory,
replacing tokens with the values defined in the init
target.
|
Internal |
compile |
Compile the java sources and put the result in the output
directory. All copies non java source files such as
.properties files, XML configuration files, ...
|
Internal |
source |
Generate a zipped file containing the full sources of the project (i.e. the whole directory structure, excluding any build generated files). | External |
javadoc |
Generates the javadoc of the project. This javadoc will be part
of the project documentation, as generated by the doc
target.
|
Internal |
doc |
Generates the full project documentation: javadoc + README files + documentation web site (using Stylebook for example). | External |
clean |
Remove any build generated files. In particular, delete the output directory. | External |
jar war ear , ...
|
Generate the project runtime. If the project is a framework, it is usually a jar. If the project is a web application, it is usually a war file. If the project is an EJB application, it is usually either one or several jars or an ear file. Etc ... | External |
tests_XXX |
Run the Cactus unit tests. | External |
You build.xml
file being in located in
build/
you should set the project
tag
basedir
attribute to '..
' so that all other paths are relative
to your project root directory. Indeed, the batch file (shell script)
that was used to bootstrap Ant is located in
<root>/build
, so '..
' is the root
directory.
Example:
<?xml version="1.0"?> [...] <project name="Cactus Sample" default="war" basedir="..">
You should define as properties all values that can be factorized and which are used often in your build file such as source locations, output locations, external jar locations and names, ... You can also define useful file patterns there (see the sample below).
A good principle is to defined any properties that depend on your
environment (such as external jar locations) in a file (let's call
it build.properties
located either where
build.xml
is located or in your home directory. The
properties defined in this file will be loaded by
build.xml
using the following code:
<!-- Give user a chance to override without editing this file (and without typing -D each time it compiles it) --> <property file="build/build.properties" /> <property file="${user.home}/build.properties" />
Here are our properties initializations:
<project name="Cactus Sample" default="war" basedir=".."> <!-- Give user a chance to override without editing this file (and without typing -D each time it compiles it) --> <property file="build/build.properties" /> <property file="${user.home}/build.properties" /> <!-- Generic project properties --> <property name="project.fullname" value="Cactus Sample"/> <property name="project.version" value="@version@"/> <property name="project.name" value="cactus-sample"/> <!-- Miscellaneous settings --> <property name="year" value="@year@"/> <property name="debug" value="on"/> <property name="optimize" value="off"/> <property name="deprecation" value="off"/> <!-- ======================================================================== Set the properties related to the source tree ======================================================================== --> <!-- Source locations for the build --> <property name="src.dir" value="src"/> <property name="src.java.dir" value="${src.dir}/share"/> <property name="src.java.servlet.dir" value="${src.dir}/servlet@servlet.api@"/> <property name="build.dir" value="build"/> <property name="etc.dir" value="${build.dir}/etc"/> <property name="lib.dir" value="lib"/> <property name="conf.dir" value="conf"/> <property name="conf.test.dir" value="conf/test"/> <property name="web.dir" value="web"/> <!-- ======================================================================== Set the properties related to the build area ======================================================================== --> <!-- Destination locations for the build (relative to the basedir as --> <!-- specified in the basedir attribute of the project tag) --> <property name="out.dir" value="out"/> <property name="out.dist.dir" value="${out.dir}/dist"/> <property name="out.lib.dir" value="${out.dir}/lib"/> <property name="out.test.dir" value="${out.dir}/test"/> <property name="out.src.dir" value="${out.dir}/src"/> <property name="out.classes.dir" value="${out.dir}/classes"/> <property name="out.doc.dir" value="${out.dir}/doc"/> <property name="out.javadoc.dir" value="${out.doc.dir}/javadoc"/> <property name="out.conf.dir" value="${out.dir}/conf"/> <!-- Names of deliverables --> <!-- The Cactus Sample war file. This is the file that should be used at runtime by end users (it excludes the test classes) --> <property name="final.war.name" value="${out.dir}/${project.name}-@servlet.api@.war"/> <!-- The full sources of Cactus Sample in a zip file --> <property name="final.src.name" value="${out.dir}/${project.name}-src-@servlet.api@.zip"/> <!-- The Cactus sample documentation in a zip file --> <property name="final.doc.name" value="${out.dir}/${project.name}-doc-@servlet.api@.zip"/> <!-- ======================================================================== Useful file patterns for targets ======================================================================== --> <!-- All source files of the projet. These source files will be copied to the destination source directory in the prepare task --> <patternset id="all.src.files"> <!-- All java files --> <include name="**/*.java"/> <!-- All doc files --> <include name="**/package.html"/> <include name="**/overview.html"/> <!-- All conf files (including test files) --> <include name="**/*.txt"/> <include name="**/*.xml"/> <include name="**/*.properties"/> </patternset> <!-- All non java files in the src directory --> <patternset id="all.nonjava.files"> <!-- All conf files (including test files) --> <include name="**/*.txt"/> <include name="**/*.xml"/> <include name="**/*.properties"/> </patternset>
Useful for initializing a timestamp (DSTAMP, TODAY, TSTAMP), defining token filters, printing some information messages, registering custom Ant tasks, ...
<!-- ======================================================================== Initialize the build. Must be called by all targets ======================================================================== --> <target name="init"> <!-- So that we can use the ${TSTAMP}, ${DSTAMP}, ... time stamps in targets, if need be --> <tstamp/> <echo message="--------- ${project.fullname} ${project.version} ---------"/> <echo message=""/> <echo message="java.class.path = ${java.class.path}"/> <echo message=""/> <echo message="java.home = ${java.home}"/> <echo message="user.home = ${user.home}"/> <echo message=""/> <echo message="basedir = ${basedir}"/> <echo message=""/> <echo message="servlet.jar = ${servlet.jar}"/> <echo message="cactus.jar = ${cactus.jar}"/> <echo message="junit.jar = ${junit.jar}"/> <echo message="cactus.ant.jar = ${cactus.ant.jar}"/> <!-- Filters --> <filter token="version" value="${project.version}"/> <filter token="year" value="${year}"/> <!-- Initialize custom Ant tasks needed for running the server tests --> <taskdef resource="cactus.tasks"> <classpath> <pathelement location="${cactus.ant.jar}"/> <pathelement path="${java.class.path}"/> </classpath> </taskdef> </target>
Display a usage message.
<!-- ======================================================================== Help on usage. List available targets ======================================================================== --> <target name="usage" depends="init"> <echo message=""/> <echo message="${project.fullname} build file"/> <echo message="------------------------------------------------------"/> <echo message=""/> <echo message=" Available targets are:"/> <echo message=""/> <echo message=" war --> generates the war file (default)"/> <echo message=" clean --> cleans up the build directory"/> <echo message=" source --> generates source zip of the project"/> <echo message=" doc --> generates the docs (javadoc, ...)"/> <echo message=" all --> do it all at once"/> <echo message=" (clean, war, source, doc)"/> <echo message=""/> <echo message=" Targets for running the tests for Servlet API 2.2:"/> <echo message=""/> <echo message=" tests_resin_12 --> run tests for Resin 1.2"/> <echo message=" tests_tomcat_32 --> run tests for Tomcat 3.2"/> <echo message=" tests_weblogic_51 --> run tests for WebLogic 5.1"/> <echo message=" tests_orion_14 --> run tests for Orion 1.4"/> <echo message=""/> </target>
This target is needed for both compiling and generating the javadoc.
It copies all the source files from their src/
directory
to the out
directory, replacing tokens in the source
code (replacing the @version@
by the version number
for example).
<!-- ======================================================================== Prepare the output directory by copying the source files into it ======================================================================== --> <target name="prepare" depends="init"> <mkdir dir="${out.src.dir}"/> <!-- Copy all source files to destination dir. Apply the filters in order to replace the tokens for the copyright year and the version --> <copy todir="${out.src.dir}" filtering="on"> <fileset dir="${src.java.dir}"> <patternset refid="all.src.files"/> </fileset> <fileset dir="${src.java.servlet.dir}"> <patternset refid="all.src.files"/> </fileset> </copy> </target>
The compile
target simply compiles the java files into
.class
files and also copies the support files that are
in src/
to the directory where the class file have been
generated.
<!-- ======================================================================== Compiles the source directory ======================================================================== --> <!-- Preparation target for the compile target --> <target name="prepare-compile" depends="prepare"> <mkdir dir="${out.classes.dir}"/> </target> <!-- Run the java compilation --> <target name="compile" depends="prepare-compile"> <javac srcdir="${out.src.dir}" destdir="${out.classes.dir}" debug="${debug}" deprecation="${deprecation}" optimize="${optimize}"> <!-- Exclude all files that are not .java source files --> <!-- All doc files --> <exclude name="**/package.html"/> <exclude name="**/overview.html"/> <!-- All conf files (including test files) --> <exclude name="**/*.txt"/> <exclude name="**/*.xml"/> <exclude name="**/*.properties"/> <classpath> <pathelement path="${java.class.path}"/> <pathelement location="${servlet.jar}"/> <pathelement location="${cactus.jar}"/> </classpath> </javac> <!-- Copies non java files that need to be in the classes directory --> <copy todir="${out.classes.dir}"> <fileset dir="${src.java.dir}"> <patternset refid="all.nonjava.files"/> </fileset> <fileset dir="${conf.test.dir}"> <include name="cactus.properties"/> </fileset> </copy> </target>
Zip up the sources for distribution.
<!-- ======================================================================== Generates source zip of the project ======================================================================== --> <target name="source" depends="prepare"> <zip zipfile="${final.src.name}" basedir="."> <exclude name="${out.dir}/**"/> <exclude name="**/*.log"/> <exclude name="**/*.bak"/> <exclude name="**/*.class"/> <exclude name="${build.dir}/build.properties"/> </zip> </target>
Generate the project's javadoc.
<!-- ======================================================================== Generate the javadoc ======================================================================== --> <!-- Preparation target for the javadoc target --> <target name="prepare-javadoc" depends="prepare"> <mkdir dir="${out.javadoc.dir}"/> </target> <!-- Generate the javadoc for the current Servlet API --> <target name="javadoc" depends="prepare-javadoc"> <javadoc sourcepath="${out.src.dir}" packagenames="org.apache.cactus.sample.*" destdir="${out.javadoc.dir}" author="true" public="true" version="true" use="true" windowtitle="${project.fullname} ${project.version} for Servlet @servlet.api@ API" doctitle="${project.fullname} ${project.version} for Servlet @servlet.api@ API" bottom="Copyright &copy; ${year} Apache Software Foundation. All Rights Reserved."> <classpath> <pathelement path="${java.class.path}"/> <pathelement location="${servlet.jar}"/> <pathelement location="${cactus.jar}"/> </classpath> </javadoc> </target>
Generate the project's documentation. It includes the javadoc, additional README files (if any) and the documentation web site
Example:
<!-- ======================================================================== Generate the full documentation ======================================================================== --> <!-- Preparation target for the doc target --> <target name="prepare-doc" depends="javadoc"> <mkdir dir="${out.doc.dir}"/> </target> <!-- Generate the documentation --> <target name="doc" depends="prepare-doc"> <!-- Create the zipped documentation --> <zip zipfile="${final.doc.name}" basedir="${out.doc.dir}"/> </target>
Removes all build generated files.
<!-- ======================================================================== Remove all build generated files ======================================================================== --> <target name="clean" depends="init"> <!-- Deletes all files ending with '~' --> <delete> <fileset dir="." includes="**/*~" defaultexcludes="no"/> </delete> <!-- Remove the out directory --> <delete dir="${out.dir}"/> <!-- Delete log files --> <delete> <fileset dir="."> <include name="**/*.log"/> </fileset> </delete> </target>
This target is useful if your project is a framework for example and
you need to deliver a jar file. We also include a manifest file
in the jar, with version information. We copy the manifest to the
output directory in order to replace the @version@
token with its value.
Example (from the Cactus build file):
<!-- ======================================================================== Create the runtime jar file ======================================================================== --> <!-- Preparation target for the jar target --> <target name="prepare-jar" depends="compile"> <mkdir dir="${out.conf.dir}"/> <mkdir dir="${out.lib.dir}"/> <!-- Copy the manifest in order to replace the version token filter --> <copy todir="${out.conf.dir}" filtering="on"> <fileset dir="${conf.dir}" > <include name="manifest"/> </fileset> </copy> </target> <!-- Generate the jar file --> <target name="jar" depends="prepare-jar"> <jar jarfile="${final.jar.name}" basedir="${out.classes.dir}" manifest="${out.conf.dir}/manifest"> <!-- Do not include test files in the runtime jar --> <exclude name="**/Test*.*"/> <exclude name="**/test*.*"/> </jar> </target>
This target is useful if you're building a web application.
We also include a manifest file in the war, with version
information. We copy the manifest to the output directory in order
to replace the @version@
token with it's value.
Example (from the Cactus sample):
<!-- ======================================================================== Create the runtime war file ======================================================================== --> <!-- Preparation target for the war target --> <target name="prepare-war" depends="compile"> <mkdir dir="${out.conf.dir}"/> <!-- Copy the manifest in order to replace the version token filter --> <copy todir="${out.conf.dir}" filtering="on"> <fileset dir="${conf.dir}" > <include name="manifest"/> </fileset> </copy> </target> <!-- Generate the war file --> <target name="war" depends="prepare-war"> <war warfile="${final.war.name}" webxml="${conf.dir}/web.xml" manifest="${out.conf.dir}/manifest"> <classes dir="${out.classes.dir}"> <!-- Do not include test files in the runtime jar --> <exclude name="**/Test*.*"/> <exclude name="**/test*.*"/> <!-- Also exclude the test cactus.properties file --> <exclude name="cactus.properties"/> </classes> <fileset dir="${web.dir}"> <exclude name="test/**"/> </fileset> </war> </target>
The tests_XXX
target is in charge of running the
Cactus unit tests. It must prepare the test environment for a
given servlet engine and package the tests, start that servlet
engine, run the tests by starting the JUnit runner and stop the
servlet engine.
See the Using Cactus with Ant tutorial for details on to do this.