PR: 4143 git-svn-id: https://svn.apache.org/repos/asf/ant/core/trunk@269893 13f79535-47bb-0310-9956-ffa450edef68master
@@ -28,6 +28,10 @@ Fixed bugs: | |||||
contain at least a manifest file, therefore it will now print a | contain at least a manifest file, therefore it will now print a | ||||
warning and do nothing. | warning and do nothing. | ||||
* <typedef> hasn't been all that useful as it couldn't be used outside | |||||
of targets (it can now) and nested "unknown" elements have always | |||||
been considered to be tasks (changed as well). | |||||
Other changes: | Other changes: | ||||
-------------- | -------------- | ||||
@@ -7,6 +7,11 @@ | |||||
<pathelement path="${java.class.path}" /> | <pathelement path="${java.class.path}" /> | ||||
</path> | </path> | ||||
<taskdef name="global" | |||||
classname="org.example.tasks.TaskdefTestContainerTask"> | |||||
<classpath refid="testclasses" /> | |||||
</taskdef> | |||||
<target name="test1"> | <target name="test1"> | ||||
<taskdef/> | <taskdef/> | ||||
</target> | </target> | ||||
@@ -51,4 +56,10 @@ | |||||
</test7> | </test7> | ||||
</target> | </target> | ||||
<target name="testGlobal"> | |||||
<global> | |||||
<echo message="worked" /> | |||||
</global> | |||||
</target> | |||||
</project> | </project> |
@@ -0,0 +1,49 @@ | |||||
<?xml version="1.0"?> | |||||
<project name="test" basedir="." default="invalid"> | |||||
<target name="invalid"> | |||||
<fail>This file should only be run via a testcase</fail> | |||||
</target> | |||||
<target name="empty"> | |||||
<typedef /> | |||||
</target> | |||||
<target name="noClassname"> | |||||
<typedef name="dummy" /> | |||||
</target> | |||||
<target name="noName"> | |||||
<typedef classname="org.example.types.TypedefTestType"> | |||||
<classpath refid="testclasses" /> | |||||
</typedef> | |||||
</target> | |||||
<target name="classNotFound"> | |||||
<typedef name="" classname="oops"/> | |||||
</target> | |||||
<path id="testclasses"> | |||||
<pathelement location="../../../../build/testcases" /> | |||||
<pathelement path="${java.class.path}" /> | |||||
</path> | |||||
<typedef name="global" | |||||
classname="org.example.types.TypedefTestType"> | |||||
<classpath refid="testclasses" /> | |||||
</typedef> | |||||
<target name="testGlobal"> | |||||
<global id="global" /> | |||||
</target> | |||||
<target name="testLocal"> | |||||
<typedef name="local" | |||||
classname="org.example.types.TypedefTestType"> | |||||
<classpath refid="testclasses" /> | |||||
</typedef> | |||||
<local id="local" /> | |||||
</target> | |||||
</project> |
@@ -350,6 +350,8 @@ public class ProjectHelper { | |||||
public void startElement(String name, AttributeList attrs) throws SAXParseException { | public void startElement(String name, AttributeList attrs) throws SAXParseException { | ||||
if (name.equals("taskdef")) { | if (name.equals("taskdef")) { | ||||
handleTaskdef(name, attrs); | handleTaskdef(name, attrs); | ||||
} else if (name.equals("typedef")) { | |||||
handleTypedef(name, attrs); | |||||
} else if (name.equals("property")) { | } else if (name.equals("property")) { | ||||
handleProperty(name, attrs); | handleProperty(name, attrs); | ||||
} else if (name.equals("target")) { | } else if (name.equals("target")) { | ||||
@@ -365,6 +367,10 @@ public class ProjectHelper { | |||||
(new TaskHandler(this, null, null, null)).init(name, attrs); | (new TaskHandler(this, null, null, null)).init(name, attrs); | ||||
} | } | ||||
private void handleTypedef(String name, AttributeList attrs) throws SAXParseException { | |||||
(new TaskHandler(this, null, null, null)).init(name, attrs); | |||||
} | |||||
private void handleProperty(String name, AttributeList attrs) throws SAXParseException { | private void handleProperty(String name, AttributeList attrs) throws SAXParseException { | ||||
(new TaskHandler(this, null, null, null)).init(name, attrs); | (new TaskHandler(this, null, null, null)).init(name, attrs); | ||||
} | } | ||||
@@ -210,10 +210,10 @@ public class Target implements TaskContainer { | |||||
} | } | ||||
} | } | ||||
void replaceTask(UnknownElement el, Task t) { | |||||
void replaceChild(UnknownElement el, Object o) { | |||||
int index = -1; | int index = -1; | ||||
while ((index = children.indexOf(el)) >= 0) { | while ((index = children.indexOf(el)) >= 0) { | ||||
children.setElementAt(t, index); | |||||
children.setElementAt(o, index); | |||||
} | } | ||||
} | } | ||||
@@ -58,22 +58,22 @@ import java.util.Vector; | |||||
/** | /** | ||||
* Wrapper class that holds all information necessary to create a task | * Wrapper class that holds all information necessary to create a task | ||||
* that did not exist when Ant started. | |||||
* or data type that did not exist when Ant started. | |||||
* | * | ||||
* @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | ||||
*/ | */ | ||||
public class UnknownElement extends Task { | public class UnknownElement extends Task { | ||||
/** | /** | ||||
* Holds the name of the task or nested child element of a task | |||||
* that hasn't been defined at parser time. | |||||
* Holds the name of the task/type or nested child element of a | |||||
* task/type that hasn't been defined at parser time. | |||||
*/ | */ | ||||
private String elementName; | private String elementName; | ||||
/** | /** | ||||
* The task after it has been loaded. | |||||
* The real object after it has been loaded. | |||||
*/ | */ | ||||
private Task realTask; | |||||
private Object realThing; | |||||
/** | /** | ||||
* Childelements, holds UnknownElement instances. | * Childelements, holds UnknownElement instances. | ||||
@@ -92,33 +92,41 @@ public class UnknownElement extends Task { | |||||
} | } | ||||
/** | /** | ||||
* creates the task instance, creates child elements, configures | |||||
* the attributes of the task. | |||||
* creates the real object instance, creates child elements, configures | |||||
* the attributes of the real object. | |||||
*/ | */ | ||||
public void maybeConfigure() throws BuildException { | public void maybeConfigure() throws BuildException { | ||||
realTask = makeTask(this, wrapper); | |||||
realThing = makeObject(this, wrapper); | |||||
wrapper.setProxy(realTask); | |||||
realTask.setRuntimeConfigurableWrapper(wrapper); | |||||
wrapper.setProxy(realThing); | |||||
if (realThing instanceof Task) { | |||||
((Task) realThing).setRuntimeConfigurableWrapper(wrapper); | |||||
} | |||||
handleChildren(realTask, wrapper); | |||||
handleChildren(realThing, wrapper); | |||||
realTask.maybeConfigure(); | |||||
target.replaceTask(this, realTask); | |||||
wrapper.maybeConfigure(project); | |||||
if (realThing instanceof Task) { | |||||
target.replaceChild(this, realThing); | |||||
} else { | |||||
target.replaceChild(this, wrapper); | |||||
} | |||||
} | } | ||||
/** | /** | ||||
* Called when the real task has been configured for the first time. | * Called when the real task has been configured for the first time. | ||||
*/ | */ | ||||
public void execute() { | public void execute() { | ||||
if (realTask == null) { | |||||
if (realThing == null) { | |||||
// plain impossible to get here, maybeConfigure should | // plain impossible to get here, maybeConfigure should | ||||
// have thrown an exception. | // have thrown an exception. | ||||
throw new BuildException("Could not create task of type: " | throw new BuildException("Could not create task of type: " | ||||
+ elementName, location); | + elementName, location); | ||||
} | } | ||||
realTask.perform(); | |||||
if (realThing instanceof Task) { | |||||
((Task) realThing).perform(); | |||||
} | |||||
} | } | ||||
/** | /** | ||||
@@ -149,7 +157,7 @@ public class UnknownElement extends Task { | |||||
Object realChild = null; | Object realChild = null; | ||||
if (parent instanceof TaskContainer) { | if (parent instanceof TaskContainer) { | ||||
realChild = makeTask(child, childWrapper); | |||||
realChild = makeTask(child, childWrapper, false); | |||||
((TaskContainer) parent).addTask((Task) realChild); | ((TaskContainer) parent).addTask((Task) realChild); | ||||
} else { | } else { | ||||
realChild = ih.createElement(project, parent, child.getTag()); | realChild = ih.createElement(project, parent, child.getTag()); | ||||
@@ -168,44 +176,62 @@ public class UnknownElement extends Task { | |||||
} | } | ||||
} | } | ||||
/** | |||||
* Creates a named task or data type - if it is a task, configure it up to the init() stage. | |||||
*/ | |||||
protected Object makeObject(UnknownElement ue, RuntimeConfigurable w) { | |||||
Object o = makeTask(ue, w, true); | |||||
if (o == null) { | |||||
o = project.createDataType(ue.getTag()); | |||||
} | |||||
if (o == null) { | |||||
throw getNotFoundException("task or type", ue.getTag()); | |||||
} | |||||
return o; | |||||
} | |||||
/** | /** | ||||
* Create a named task and configure it up to the init() stage. | * Create a named task and configure it up to the init() stage. | ||||
*/ | */ | ||||
protected Task makeTask(UnknownElement ue, RuntimeConfigurable w) { | |||||
protected Task makeTask(UnknownElement ue, RuntimeConfigurable w, | |||||
boolean onTopLevel) { | |||||
Task task = project.createTask(ue.getTag()); | Task task = project.createTask(ue.getTag()); | ||||
if (task == null) { | |||||
String lSep = System.getProperty("line.separator"); | |||||
String msg = "Could not create task of type: " + elementName | |||||
+ "." + lSep | |||||
+ "Ant could not find the task or a class this" + lSep | |||||
+ "task relies upon." + lSep | |||||
+ "Common solutions are to use taskdef to declare" + lSep | |||||
+ "your task, or, if this is an optional task," + lSep | |||||
+ "to put the optional.jar and all required libraries of" +lSep | |||||
+ "this task in the lib directory of" + lSep | |||||
+ "your ant installation (ANT_HOME)."; | |||||
throw new BuildException(msg, location); | |||||
if (task == null && !onTopLevel) { | |||||
throw getNotFoundException("task", ue.getTag()); | |||||
} | } | ||||
task.setLocation(getLocation()); | |||||
String id = w.getAttributes().getValue("id"); | |||||
if (id != null) { | |||||
project.addReference(id, task); | |||||
if (task != null) { | |||||
task.setLocation(getLocation()); | |||||
// UnknownElement always has an associated target | |||||
task.setOwningTarget(target); | |||||
task.init(); | |||||
} | } | ||||
// UnknownElement always has an associated target | |||||
task.setOwningTarget(target); | |||||
task.init(); | |||||
return task; | return task; | ||||
} | } | ||||
protected BuildException getNotFoundException(String what, | |||||
String elementName) { | |||||
String lSep = System.getProperty("line.separator"); | |||||
String msg = "Could not create " + what + " of type: " + elementName | |||||
+ "." + lSep | |||||
+ "Ant could not find the task or a class this" + lSep | |||||
+ "task relies upon." + lSep | |||||
+ "Common solutions are to use taskdef to declare" + lSep | |||||
+ "your task, or, if this is an optional task," + lSep | |||||
+ "to put the optional.jar and all required libraries of" +lSep | |||||
+ "this task in the lib directory of" + lSep | |||||
+ "your ant installation (ANT_HOME)."; | |||||
return new BuildException(msg, location); | |||||
} | |||||
/** | /** | ||||
* Get the name to use in logging messages. | * Get the name to use in logging messages. | ||||
* | * | ||||
* @return the name to use in logging messages. | * @return the name to use in logging messages. | ||||
*/ | */ | ||||
public String getTaskName() { | public String getTaskName() { | ||||
return realTask == null ? super.getTaskName() : realTask.getTaskName(); | |||||
return realThing == null || !(realThing instanceof Task) ? | |||||
super.getTaskName() : ((Task) realThing).getTaskName(); | |||||
} | } | ||||
}// UnknownElement | }// UnknownElement |
@@ -100,4 +100,8 @@ public class TaskdefTest extends TaskdefsTest { | |||||
public void test7() { | public void test7() { | ||||
expectLog("test7", "worked"); | expectLog("test7", "worked"); | ||||
} | } | ||||
public void testGlobal() { | |||||
expectLog("testGlobal", "worked"); | |||||
} | |||||
} | } |
@@ -0,0 +1,107 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2001 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.apache.tools.ant.taskdefs; | |||||
import org.apache.tools.ant.Project; | |||||
/** | |||||
* @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||||
* @version $Revision$ | |||||
*/ | |||||
public class TypedefTest extends TaskdefsTest { | |||||
public TypedefTest(String name) { | |||||
super(name); | |||||
} | |||||
public void setUp() { | |||||
configureProject("src/etc/testcases/taskdefs/typedef.xml"); | |||||
} | |||||
public void testEmpty() { | |||||
expectBuildException("empty", "required argument not specified"); | |||||
} | |||||
public void testNoName() { | |||||
expectBuildException("noName", "required argument not specified"); | |||||
} | |||||
public void testNoClassname() { | |||||
expectBuildException("noClassname", "required argument not specified"); | |||||
} | |||||
public void testClassNotFound() { | |||||
expectBuildException("classNotFound", | |||||
"classname specified doesn't exist"); | |||||
} | |||||
public void testGlobal() { | |||||
expectLog("testGlobal", ""); | |||||
Object ref = project.getReferences().get("global"); | |||||
assertNotNull("ref is not null", ref); | |||||
assertEquals("org.example.types.TypedefTestType", | |||||
ref.getClass().getName()); | |||||
} | |||||
public void testLocal() { | |||||
expectLog("testLocal", | |||||
"Overriding previous definition of reference to local"); | |||||
Object ref = project.getReferences().get("local"); | |||||
assertNotNull("ref is not null", ref); | |||||
assertEquals("org.example.types.TypedefTestType", | |||||
ref.getClass().getName()); | |||||
} | |||||
} |
@@ -0,0 +1,61 @@ | |||||
/* | |||||
* The Apache Software License, Version 1.1 | |||||
* | |||||
* Copyright (c) 2001 The Apache Software Foundation. All rights | |||||
* reserved. | |||||
* | |||||
* Redistribution and use in source and binary forms, with or without | |||||
* modification, are permitted provided that the following conditions | |||||
* are met: | |||||
* | |||||
* 1. Redistributions of source code must retain the above copyright | |||||
* notice, this list of conditions and the following disclaimer. | |||||
* | |||||
* 2. Redistributions in binary form must reproduce the above copyright | |||||
* notice, this list of conditions and the following disclaimer in | |||||
* the documentation and/or other materials provided with the | |||||
* distribution. | |||||
* | |||||
* 3. The end-user documentation included with the redistribution, if | |||||
* any, must include the following acknowlegement: | |||||
* "This product includes software developed by the | |||||
* Apache Software Foundation (http://www.apache.org/)." | |||||
* Alternately, this acknowlegement may appear in the software itself, | |||||
* if and wherever such third-party acknowlegements normally appear. | |||||
* | |||||
* 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||||
* Foundation" must not be used to endorse or promote products derived | |||||
* from this software without prior written permission. For written | |||||
* permission, please contact apache@apache.org. | |||||
* | |||||
* 5. Products derived from this software may not be called "Apache" | |||||
* nor may "Apache" appear in their names without prior written | |||||
* permission of the Apache Group. | |||||
* | |||||
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||||
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||||
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||||
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||||
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||||
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||||
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||||
* SUCH DAMAGE. | |||||
* ==================================================================== | |||||
* | |||||
* This software consists of voluntary contributions made by many | |||||
* individuals on behalf of the Apache Software Foundation. For more | |||||
* information on the Apache Software Foundation, please see | |||||
* <http://www.apache.org/>. | |||||
*/ | |||||
package org.example.types; | |||||
import org.apache.tools.ant.ProjectComponent; | |||||
public class TypedefTestType extends ProjectComponent { | |||||
} | |||||