Tutorial: Writing Tasks

Jan Matèrne, 2003-09-01

This document provides a tutorial for writing tasks.

Set up the build environment

Ant builds itself, we are using Ant too (why we would write a task if not? :-) therefore we should use Ant for our build.

We choose a directory as root directory. All things will be done here if I say nothing different. I will reference this directory as root-directory of our project. In this root-directory we create a text file names build.xml. What should Ant do for us?

So the buildfile contains three targets.

{{{ <?xml version="1.0" encoding="ISO-8859-1"?>

This buildfile uses often the same value (src, classes, MyTask.jar), so we should rewrite that using <property>s. On second there are some handicaps: <javac> requires that the destination directory exists; a call of "clean" with a non existing classes directory will fail; "jar" requires the execution of some steps before. So the refactored code is:

{{{ <?xml version="1.0" encoding="ISO-8859-1"?>

ant.project.name is one of the build-in properties [1] of Ant.

Write the Task

Now we write the simplest Task - a HelloWorld-Task (what else?). Create a text file HelloWorld.java in the src-directory with:

{{{ public class HelloWorld {

and we can compile and jar it with  ant  (default target is "jar" and via its depends-clause the "compile" is executed before).

Use the Task

But after creating the jar we want to use our new Task. Therefore we need a new target "use". Before we can use our new task we have to declare it with <taskdef> [2]. And for easier process we change the default clause:

{{{ <?xml version="1.0" encoding="ISO-8859-1"?>

Important is the classpath-attribute. Ant searches in its /lib directory for tasks and our task isn´t there. So we have to provide the right location.

Now we can type in  ant  and all should work ...

{{{ Buildfile: build.xml

Integration with TaskAdapter

Our class has nothing to do with Ant. It extends no superclass and implements no interface. How does Ant know to integrate? Via name convention: our class provides a method with signature public void execute(). This class is wrapped by Ant´s  org.apache.tools.ant.TaskAdapter  which is a task and uses reflection for setting a reference to the project and calling the execute() method.

Setting a reference to the project? Could be interesting. The Project class gives us some nice abilities: access to Ant´s logging facilities getting and setting properties and much more. So we try to use that class:

{{{ import org.apache.tools.ant.Project;

and the execution with ant will show us the expected

{{{ use:

Deriving from Ant´s Task

Ok, that works ... But usually you will extend  org.apache.tools.ant.Task . That class is integrated in Ant, get´s the project-reference, provides documentation fiels, provides easier access to the logging facility and (very useful) gives you the exact location where in the buildfile this task instance is used.

Oki-doki - let´s us use some of these:

{{{ import org.apache.tools.ant.Task;

which gives us when running {{{ use:

Attributes

Now we want to specify the text of our message (it seems that we are rewriting the <echo/> task :-). First we well do that with an attribute. It is very easy - for each attribute provide a public void  set<attributename>(<type> newValue)  method and Ant will do the rest via reflection.

{{{ import org.apache.tools.ant.Task;

Oh, what´s that in execute()? Throw a BuildException? Yes, that´s the usual way to show Ant that something important is missed and complete build should fail. The string provided there is written as build-failes-message. Here it´s necessary because the log() method can´t handle a null value as parameter and throws a NullPointerException. (Of course you can initialize the message with a default string.)

After that we have to modify our buildfile:

{{{ <target name="use" description="Use the Task" depends="jar">

That´s all.

Some background for working with attributes: Ant supports any of these datatypes as arguments of the set-method:

Before calling the set-method all properties are resolved. So a <helloworld message="${msg}"/> would not set the message string to "${msg}" if there is a property "msg" with a set value.

Nested Text

Maybe you have used the <echo> task in a way like <echo>Hello World</echo>. For that you have to provide a public void addText(String text) method.

{{{ ...

But here properties are not resolved! For resolving properties we have to use Project´s  replaceProperties(String propname) : String  method which takes the property name as argument and returns its value (or ${propname} if not set).

Nested Elements

There are several ways for inserting the ability of handling nested elements. See the Manual [4] for other. We use the first way of the three described ways. There are several steps for that:

{{{ import java.util.Vector;

Then we can use the new nested element. But where is xml-name for that defined? The mapping XML-name : classname is defined in the factory method:  public classname createXML-name() . Therefore we write in the buildfile

{{{ <helloworld>

Our task in a little more complex version

For recapitulation now a little refactored buildfile:

{{{ <?xml version="1.0" encoding="ISO-8859-1"?>

And the code of the task:

{{{ import org.apache.tools.ant.Task;

And it works:

{{{ C:\tmp\anttests\MyFirstTask>ant

Next step: test ...

Test the Task

We have written a test already: the use.* tasks in the buildfile. But its difficult to test that automatically. Common (and in Ant) used is JUnit for that. For testing tasks Ant provides a baseclass  org.apache.tools.ant.BuildFileTest . This class extends  junit.framework.TestCase  and can therefore be integrated into the unit tests. But this class provides some for testing tasks useful methods: initialize Ant, load a buildfile, execute targets, expecting BuildExceptions with a specified text, expect a special text in the output log ...

In Ant it is usual that the testcase has the same name as the task with a prepending Test, therefore we will create a file HelloWorldTest.java. Because we have a very small project we can put this file into src directory (Ant´s own testclasses are in /src/testcases/...). Because we have already written our tests for "hand-test" we can use that for automatic tests, too. But there is one little problem we have to solve: all test supporting classes are not part of the binary distribution of Ant. So you can build the special jar file from source distro with target "test-jar" or you can download a nightly build from http://gump.covalent.net/jars/latest/ant/ant-testutil.jar [5].

For executing the test and creating a report we need the optional tasks <junit> and <junitreport>. So we add to the buildfile:

{{{ ...

Back to the src/HelloWorldTest.java. We create a class extending BuildFileTest with String-constructor (JUnit-standard), a setUp() method initializing Ant and for each testcase (targets use.*) a testXX() method invoking that target.

{{{ import org.apache.tools.ant.BuildFileTest;

When starting  ant  we´ll get a short message to STDOUT and a nice HTML-report.

{{{ C:\tmp\anttests\MyFirstTask>ant

Resources

This tutorial and its resources are available via BugZilla [6]. The ZIP provided there contains

Used Links: {{{ [1] http://ant.apache.org/manual/using.html#built-in-props

AntTutorialWritingTasks (last edited 2009-09-20 23:31:28 by localhost)