| <!-- |
| Licensed to the Apache Software Foundation (ASF) under one or more |
| contributor license agreements. See the NOTICE file distributed with |
| this work for additional information regarding copyright ownership. |
| The ASF licenses this file to You under the Apache License, Version 2.0 |
| (the "License"); you may not use this file except in compliance with |
| the License. You may obtain a copy of the License at |
| |
| http://www.apache.org/licenses/LICENSE-2.0 |
| |
| Unless required by applicable law or agreed to in writing, software |
| distributed under the License is distributed on an "AS IS" BASIS, |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| See the License for the specific language governing permissions and |
| limitations under the License. |
| --> |
| <html> |
| |
| <head> |
| <meta http-equiv="Content-Language" content="en-us"> |
| <link rel="stylesheet" type="text/css" href="../stylesheets/style.css"> |
| <title>Scriptdef Task</title> |
| </head> |
| |
| <body> |
| |
| <h2><a name="script">Scriptdef</a></h2> |
| <h3>Description</h3> |
| <p>Scriptdef can be used to define an Apache Ant task using a scripting language. Ant |
| scripting languages supported by |
| <a href="http://jakarta.apache.org/bsf" target="_top">Apache BSF</a> |
| or |
| <a href="https://scripting.dev.java.net">JSR 223</a> |
| may be |
| used to define the script. Scriptdef provides a mechanism to encapsulate |
| control logic from a build within an Ant task minimizing the need for |
| providing control style tasks in Ant itself. Complex logic can be made |
| available while retaining the simple structure of an Ant build file. Scriptdef |
| is also useful for prototyping new custom tasks. Certainly as the complexity |
| of the script increases it would be better to migrate the task definition |
| into a Java based custom task. |
| </p> |
| |
| <p><b>Note:</b> This task depends on external libraries not included in the |
| Ant distribution. See |
| <a href="../install.html#librarydependencies">Library Dependencies</a> |
| for more information.</p> |
| |
| |
| |
| <p>The attributes and nested elements supported by the task may be defined |
| using <code><attribute></code> and <code><element></code> nested elements. These are |
| available to the script that implements the task as two collection style |
| script variables <code>attributes</code> and <code>elements</code>. The |
| elements in the <code>attributes</code> collection may be accessed by the |
| attribute name. The <code>elements</code> collection is accessed by the nested |
| element name. This will return a list of all instances of the nested element. |
| The instances in this list may be accessed by an integer index. |
| </p> |
| |
| <p><b>Note:</b> Ant will turn all attribute and element names into all |
| lowercase names, so even if you use name="SomeAttribute", you'll have |
| to use "someattribute" to retrieve the attribute's value from the |
| <code>attributes</code> collection.</p> |
| |
| <p>The name "self" (<i>since Ant 1.6.3</i>) is a pre-defined reference to the |
| script def task instance. |
| It can be used for logging, or for integration with the rest of |
| ant. the <code>self.text attribute</code> contains |
| any nested text passed to the script</p> |
| |
| <p>If an attribute or element is not passed in, |
| then <code>attributes.get()</code> or <code>elements.get()</code> will |
| return null. It is up to the script to perform any checks and validation. |
| <code>self.fail(String message)</code>can be used to raise a |
| <code>BuildException</code>. |
| </p> |
| |
| |
| <p>The name "project" is a pre-defined reference to the Ant Project. For |
| more information on writing scripts, please refer to the |
| <a href="script.html"><code><script></code></a> task |
| </p> |
| |
| |
| <h3>Parameters</h3> |
| <table border="1" cellpadding="2" cellspacing="0"> |
| <tr> |
| <td valign="top"><b>Attribute</b></td> |
| <td valign="top"><b>Description</b></td> |
| <td align="center" valign="top"><b>Required</b></td> |
| </tr> |
| <tr> |
| <td valign="top">name</td> |
| <td valign="top">the name of the task to be created using the script</td> |
| <td valign="top" align="center">Yes</td> |
| </tr> |
| <tr> |
| <td valign="top">language</td> |
| <td valign="top">The programming language the script is written in. |
| Must be a supported Apache BSF or JSR 223 language</td> |
| <td valign="top" align="center">Yes</td> |
| </tr> |
| <tr> |
| <td valign="top">manager</td> |
| <td valign="top"> |
| The script engine manager to use. |
| See the <a href="../Tasks/script.html">script</a> task |
| for using this attribute. |
| </td> |
| <td valign="top" align="center">No - default is "auto"</td> |
| </tr> |
| <tr> |
| <td valign="top">src</td> |
| <td valign="top">The location of the script as a file, if not inline</td> |
| <td valign="top" align="center">No</td> |
| </tr> |
| <tr> |
| <td valign="top">uri</td> |
| <td valign="top"> |
| The XML namespace uri that this definition should live in. |
| </td> |
| <td valign="top" align="center">No</td> |
| </tr> |
| <tr> |
| <td valign="top">classpath</td> |
| <td valign="top"> |
| The classpath to pass into the script. |
| </td> |
| <td align="center" valign="top">No</td> |
| </tr> |
| <tr> |
| <td valign="top">classpathref</td> |
| <td valign="top">The classpath to use, given as a |
| <a href="../using.html#references">reference</a> to a path defined elsewhere. |
| <td align="center" valign="top">No</td> |
| </tr> |
| <tr> |
| <td valign="top">loaderRef</td> |
| <td valign="top">the name of the loader that is |
| used to load the script, constructed from the specified |
| classpath. This allows multiple script definitions |
| to reuse the same class loader. |
| </td> |
| <td align="center" valign="top">No</td> |
| </tr> |
| </table> |
| |
| <h3>Nested elements</h3> |
| <h4>attribute</h4> |
| <table border="1" cellpadding="2" cellspacing="0"> |
| <tr> |
| <td valign="top"><b>Attribute</b></td> |
| <td valign="top"><b>Description</b></td> |
| <td align="center" valign="top"><b>Required</b></td> |
| </tr> |
| <tr> |
| <td valign="top">name</td> |
| <td valign="top">the name of the attribute</td> |
| <td valign="top" align="center">Yes</td> |
| </tr> |
| </table> |
| |
| <h4>element</h4> |
| <table border="1" cellpadding="2" cellspacing="0"> |
| <tr> |
| <td valign="top"><b>Attribute</b></td> |
| <td valign="top"><b>Description</b></td> |
| <td align="center" valign="top"><b>Required</b></td> |
| </tr> |
| <tr> |
| <td valign="top">name</td> |
| <td valign="top">the name of the nested element to be supported by the |
| task defined by the script</td> |
| <td valign="top" align="center">Yes</td> |
| </tr> |
| <tr> |
| <td valign="top">classname</td> |
| <td valign="top">the classname of the class to be used for the nested element. |
| This specifies the class directly and is an alternative to specifying |
| the Ant type name.</td> |
| <td valign="top" align="center">No</td> |
| </tr> |
| <tr> |
| <td valign="top">type</td> |
| <td valign="top">This is the name of an Ant task or type which is to |
| be used when this element is to be created. This is an alternative |
| to specifying the class name directly. If the type is in a namespace, |
| the URI and a : must be prefixed to the type. For example |
| <code>type="antlib:example.org:newtype"</code></td> |
| <td valign="top" align="center">No</td> |
| </tr> |
| <tr> |
| <td valign="top">any resource or resource collection</td> |
| <td valign="top">Since Ant1.7.1, this task can load scripts |
| from any resource supplied as a nested element. when </td> |
| <td valign="top" align="center">No</td> |
| </tr> |
| |
| </table> |
| |
| <h4>classpath</h4> |
| <p> |
| See the <a href="../Tasks/script.html">script</a> task |
| for using this nested element. |
| </p> |
| |
| |
| <h3>Examples</h3> |
| |
| <p> |
| The following definition creates a task which supports an attribute called |
| attr and two nested elements, one being a fileset and the other a path. When |
| executed, the resulting task logs the value of the attribute and the basedir |
| of the first fileset. |
| </p> |
| |
| <pre> |
| <scriptdef name="scripttest" language="javascript"> |
| <attribute name="attr1"/> |
| <element name="fileset" type="fileset"/> |
| <element name="path" type="path"/> |
| <![CDATA[ |
| |
| self.log("Hello from script"); |
| self.log("Attribute attr1 = " + attributes.get("attr1")); |
| self.log("First fileset basedir = " |
| + elements.get("fileset").get(0).getDir(project)); |
| |
| ]]> |
| </scriptdef> |
| |
| <scripttest attr1="test"> |
| <path> |
| <pathelement location="src"/> |
| </path> |
| <fileset dir="src"/> |
| <fileset dir="main"/> |
| </scripttest> |
| </pre> |
| |
| <p> |
| The following variation on the above script lists the number of fileset elements |
| and iterates through them |
| </p> |
| <pre> |
| <scriptdef name="scripttest2" language="javascript"> |
| <element name="fileset" type="fileset"/> |
| <![CDATA[ |
| filesets = elements.get("fileset"); |
| self.log("Number of filesets = " + filesets.size()); |
| for (i = 0; i < filesets.size(); ++i) { |
| self.log("fileset " + i + " basedir = " |
| + filesets.get(i).getDir(project)); |
| } |
| ]]> |
| </scriptdef> |
| |
| <scripttest2> |
| <fileset dir="src"/> |
| <fileset dir="main"/> |
| </scripttest2> |
| </pre> |
| |
| <p> |
| When a script has a syntax error, the scriptdef name will be listed in the |
| error. For example in the above script, removing the closing curly bracket |
| would result in this error |
| </p> |
| |
| <p><code>build.xml:15: SyntaxError: missing } in compound |
| statement (scriptdef <code><scripttest2></code>; line 10)</code></p> |
| |
| <p> |
| Script errors are only detected when a script task is actually executed. |
| </p> |
| <p> |
| The next example does uses nested text in Jython. It also declares |
| the script in a new xml namespace, which must be used to refer to |
| the task. Declaring scripts in a new namespace guarantees that Ant will |
| not create a task of the same (namespace,localname) name pair. |
| </p> |
| |
| <pre> |
| <target name="echo-task-jython"> |
| <scriptdef language="jython" |
| name="echo" |
| uri="http://example.org/script"> |
| <![CDATA[ |
| self.log("text: " +self.text) |
| ]]> |
| </scriptdef> |
| </target> |
| |
| <target name="testEcho" depends="echo-task-jython" |
| xmlns:s="http://example.org/script"> |
| <s:echo>nested text</s:echo> |
| </target> |
| </pre> |
| |
| The next example shows the use of <classpath> and |
| "loaderref" to get access to the beanshell jar. |
| <pre> |
| <scriptdef name="b1" language="beanshell" |
| loaderref="beanshell-ref"> |
| <attribute name="a"/> |
| <classpath |
| path="${user.home}/scripting/beanshell/bsh-1.3b1.jar"/> |
| self.log("attribute a is " + attributes.get("a")); |
| </scriptdef> |
| |
| <scriptdef name="b2" language="beanshell" |
| loaderref="beanshell-ref"> |
| <attribute name="a2"/> |
| self.log("attribute a2 is " + attributes.get("a2")); |
| </scriptdef> |
| |
| <b1 a="this is an 'a'"/> |
| <b2 a2="this is an 'a2' for b2"/> |
| </pre> |
| <h3>Testing Scripts</h3> |
| |
| <p> |
| The easiest way to test scripts is to use the |
| <a href="http://ant.apache.org/antlibs/antunit/">AntUnit</a> ant library. |
| This will run all targets in a script that begin with "test" (and their dependencies). </p> |
| |
| |
| |
| |
| </body> |
| </html> |