git-svn-id: https://svn.apache.org/repos/asf/ant/core/trunk@272015 13f79535-47bb-0310-9956-ffa450edef68master
| @@ -201,7 +201,8 @@ | |||||
| useexternalfile="yes" | useexternalfile="yes" | ||||
| sourcepath="${java.dir}/antcore:${java.dir}/init:${java.dir}/common:${java.dir}/cli:${java.dir}/start" | sourcepath="${java.dir}/antcore:${java.dir}/init:${java.dir}/common:${java.dir}/cli:${java.dir}/start" | ||||
| destdir="${javadocs.dir}" | destdir="${javadocs.dir}" | ||||
| author="true" private ="true" | |||||
| author="true" | |||||
| private ="true" | |||||
| version="true" | version="true" | ||||
| windowtitle="Mutant API" | windowtitle="Mutant API" | ||||
| doctitle="Mutant"> | doctitle="Mutant"> | ||||
| @@ -0,0 +1,176 @@ | |||||
| /* | |||||
| * The Apache Software License, Version 1.1 | |||||
| * | |||||
| * Copyright (c) 2002 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.ant.antcore.execution; | |||||
| import java.lang.reflect.Constructor; | |||||
| import java.lang.reflect.InvocationTargetException; | |||||
| import java.lang.reflect.Method; | |||||
| import org.apache.ant.common.antlib.Converter; | |||||
| import org.apache.ant.common.util.ExecutionException; | |||||
| /** | |||||
| * AttributeSetters are created at introspection time for each | |||||
| * setter method a class provides and for which a conversion from a | |||||
| * String value is available. | |||||
| * | |||||
| * @author Conor MacNeill | |||||
| * @created 19 January 2002 | |||||
| */ | |||||
| public class AttributeSetter { | |||||
| /** The method that will perform the setting */ | |||||
| private Method method; | |||||
| /** | |||||
| * A converter to convert the string value to a value to be given to | |||||
| * the setter method | |||||
| */ | |||||
| private Converter converter; | |||||
| /** | |||||
| * A constructor used to create the string value to an object to be used | |||||
| * by the setter | |||||
| */ | |||||
| private Constructor valueConstructor; | |||||
| /** The depth of the setter in the class hierarchy */ | |||||
| private int depth; | |||||
| /** | |||||
| * Create a setter which just uses string values | |||||
| * | |||||
| * @param method the method to be invoked. | |||||
| * @param depth the depth of this method declaraion in the class hierarchy. | |||||
| */ | |||||
| public AttributeSetter(Method method, int depth) { | |||||
| this.method = method; | |||||
| this.depth = depth; | |||||
| } | |||||
| /** | |||||
| * Create a setter which just uses string values | |||||
| * | |||||
| * @param method the method to be invoked. | |||||
| * @param depth the depth of this method declaraion in the class hierarchy. | |||||
| * @param converter a converter to convert string values into instances of | |||||
| * the type expected by the method. | |||||
| */ | |||||
| public AttributeSetter(Method method, int depth, Converter converter) { | |||||
| this(method, depth); | |||||
| this.converter = converter; | |||||
| } | |||||
| /** | |||||
| * Create a setter which just uses string values | |||||
| * | |||||
| * @param method the method to be invoked. | |||||
| * @param depth the depth of this method declaraion in the class hierarchy. | |||||
| * @param valueConstructor an object constructor used to convert string | |||||
| * values into instances of the type expected by the method. | |||||
| */ | |||||
| public AttributeSetter(Method method, int depth, | |||||
| Constructor valueConstructor) { | |||||
| this(method, depth); | |||||
| this.valueConstructor = valueConstructor; | |||||
| } | |||||
| /** | |||||
| * Set the attribute value on an object | |||||
| * | |||||
| * @param obj the object on which the set method is to be invoked | |||||
| * @param stringValue the string representation of the value | |||||
| * @exception InvocationTargetException if the method cannot be | |||||
| * invoked | |||||
| * @exception IllegalAccessException if the method cannot be invoked | |||||
| * @exception ExecutionException if the conversion of the value | |||||
| * fails | |||||
| */ | |||||
| void set(Object obj, String stringValue) | |||||
| throws InvocationTargetException, IllegalAccessException, | |||||
| ExecutionException { | |||||
| Object value = null; | |||||
| if (converter != null) { | |||||
| Class type = getType(); | |||||
| value = converter.convert(stringValue, type); | |||||
| } else if (valueConstructor != null) { | |||||
| try { | |||||
| value = valueConstructor.newInstance(new String[]{stringValue}); | |||||
| } catch (InstantiationException e) { | |||||
| throw new ExecutionException(e); | |||||
| } | |||||
| } else { | |||||
| value = stringValue; | |||||
| } | |||||
| method.invoke(obj, new Object[]{value}); | |||||
| } | |||||
| /** | |||||
| * Get the declaration depth of this setter. | |||||
| * | |||||
| * @return the attribute setter's declaration depth. | |||||
| */ | |||||
| public int getDepth() { | |||||
| return depth; | |||||
| } | |||||
| /** | |||||
| * Get the type expected by this setter's method | |||||
| * | |||||
| * @return a Class instance being the type this setter's method accepts. | |||||
| */ | |||||
| public Class getType() { | |||||
| return method.getParameterTypes()[0]; | |||||
| } | |||||
| } | |||||
| @@ -54,6 +54,7 @@ | |||||
| package org.apache.ant.antcore.execution; | package org.apache.ant.antcore.execution; | ||||
| import java.lang.reflect.Method; | import java.lang.reflect.Method; | ||||
| import java.util.Map; | import java.util.Map; | ||||
| import java.util.HashMap; | |||||
| /** | /** | ||||
| * Introspects a class and builds a reflector for setting values on | * Introspects a class and builds a reflector for setting values on | ||||
| @@ -66,6 +67,27 @@ public class ClassIntrospector { | |||||
| /** The reflector that this introspector populates */ | /** The reflector that this introspector populates */ | ||||
| private Reflector reflector; | private Reflector reflector; | ||||
| /** | |||||
| * A Map which maps the classnames to their depth in the class hiearchy, | |||||
| * with the current class being depth=0 | |||||
| */ | |||||
| private Map classDepth = new HashMap(); | |||||
| /** | |||||
| * Determine the class hierarchy depths for the given class. | |||||
| * | |||||
| * @param bean the class for which the class depths will be determined. | |||||
| */ | |||||
| private void getDepths(Class bean) { | |||||
| Class currentClass = bean; | |||||
| int index = 0; | |||||
| while (currentClass != null) { | |||||
| classDepth.put(currentClass, new Integer(index++)); | |||||
| currentClass = currentClass.getSuperclass(); | |||||
| } | |||||
| } | |||||
| /** | /** | ||||
| * Create a introspector for the bean | * Create a introspector for the bean | ||||
| * | * | ||||
| @@ -75,6 +97,7 @@ public class ClassIntrospector { | |||||
| */ | */ | ||||
| public ClassIntrospector(final Class bean, Map converters) { | public ClassIntrospector(final Class bean, Map converters) { | ||||
| reflector = new Reflector(); | reflector = new Reflector(); | ||||
| getDepths(bean); | |||||
| Method[] methods = bean.getMethods(); | Method[] methods = bean.getMethods(); | ||||
| for (int i = 0; i < methods.length; i++) { | for (int i = 0; i < methods.length; i++) { | ||||
| @@ -93,8 +116,9 @@ public class ClassIntrospector { | |||||
| && returnType.equals(Void.TYPE) | && returnType.equals(Void.TYPE) | ||||
| && args.length == 1 | && args.length == 1 | ||||
| && !args[0].isArray()) { | && !args[0].isArray()) { | ||||
| reflector.addAttributeMethod(m, getPropertyName(name, "set"), | |||||
| converters); | |||||
| Integer depth = (Integer)classDepth.get(m.getDeclaringClass()); | |||||
| reflector.addAttributeMethod(m, depth.intValue(), | |||||
| getPropertyName(name, "set"), converters); | |||||
| } else if (name.startsWith("addConfigured") | } else if (name.startsWith("addConfigured") | ||||
| && name.length() > 13 | && name.length() > 13 | ||||
| && returnType.equals(Void.TYPE) | && returnType.equals(Void.TYPE) | ||||
| @@ -71,31 +71,6 @@ import org.apache.ant.common.util.ExecutionException; | |||||
| */ | */ | ||||
| public class Reflector implements Setter { | public class Reflector implements Setter { | ||||
| /** | |||||
| * AttributeSetter classes are created at introspection time for each | |||||
| * setter method a class provides and for which a conversion from a | |||||
| * String value is available. | |||||
| * | |||||
| * @author Conor MacNeill | |||||
| * @created 19 January 2002 | |||||
| */ | |||||
| private interface AttributeSetter { | |||||
| /** | |||||
| * Set the attribute value on an object | |||||
| * | |||||
| * @param obj the object on which the set method is to be invoked | |||||
| * @param value the string representation of the value | |||||
| * @exception InvocationTargetException if the method cannot be | |||||
| * invoked | |||||
| * @exception IllegalAccessException if the method cannot be invoked | |||||
| * @exception ExecutionException if the conversion of the value | |||||
| * fails | |||||
| */ | |||||
| void set(Object obj, String value) | |||||
| throws InvocationTargetException, IllegalAccessException, | |||||
| ExecutionException; | |||||
| } | |||||
| /** | /** | ||||
| * An element adder is used to add an instance of an element to an of an | * An element adder is used to add an instance of an element to an of an | ||||
| * object. The object being added will have been fully configured by Ant | * object. The object being added will have been fully configured by Ant | ||||
| @@ -359,6 +334,36 @@ public class Reflector implements Setter { | |||||
| return elementAdders.containsKey(elementName.toLowerCase()); | return elementAdders.containsKey(elementName.toLowerCase()); | ||||
| } | } | ||||
| /** | |||||
| * Add an attribute setter for the given property. The setter will only | |||||
| * be added if it does not override a higher priorty setter | |||||
| * | |||||
| * @param attributeName the name of the attribute that the setter operates | |||||
| * upon. | |||||
| * @param setter the AttribnuteSetter instance to use. | |||||
| */ | |||||
| private void addAttributeSetter(String attributeName, | |||||
| AttributeSetter setter) { | |||||
| String name = attributeName.toLowerCase(); | |||||
| AttributeSetter currentSetter | |||||
| = (AttributeSetter)attributeSetters.get(name); | |||||
| if (currentSetter != null) { | |||||
| // there is a setter, is it lower down in the class hierarchy | |||||
| int currentDepth = currentSetter.getDepth(); | |||||
| if (currentDepth < setter.getDepth()) { | |||||
| return; | |||||
| } else if (currentDepth == setter.getDepth()) { | |||||
| // now check the types | |||||
| Class currentType = currentSetter.getType(); | |||||
| if (currentType != String.class) { | |||||
| return; | |||||
| } | |||||
| } | |||||
| } | |||||
| attributeSetters.put(name, setter); | |||||
| } | |||||
| /** | /** | ||||
| * Determine if the class associated with this reflector supports a | * Determine if the class associated with this reflector supports a | ||||
| * particular nested element | * particular nested element | ||||
| @@ -375,51 +380,33 @@ public class Reflector implements Setter { | |||||
| * Add a method to the reflector for setting an attribute value | * Add a method to the reflector for setting an attribute value | ||||
| * | * | ||||
| * @param m the method, obtained by introspection. | * @param m the method, obtained by introspection. | ||||
| * @param depth the depth of this method's declaration in the class | |||||
| * hierarchy | |||||
| * @param propertyName the property name the method will set. | * @param propertyName the property name the method will set. | ||||
| * @param converters A map of converter classes used to convert strings | * @param converters A map of converter classes used to convert strings | ||||
| * to different types. | * to different types. | ||||
| */ | */ | ||||
| public void addAttributeMethod(final Method m, String propertyName, | |||||
| Map converters) { | |||||
| final Class type = m.getParameterTypes()[0]; | |||||
| public void addAttributeMethod(Method m, int depth, | |||||
| String propertyName, Map converters) { | |||||
| Class type = m.getParameterTypes()[0]; | |||||
| if (converters != null && converters.containsKey(type)) { | if (converters != null && converters.containsKey(type)) { | ||||
| // we have a converter to use to convert the String | // we have a converter to use to convert the String | ||||
| // value into something the set method expects. | // value into something the set method expects. | ||||
| Converter converter = (Converter)converters.get(type); | Converter converter = (Converter)converters.get(type); | ||||
| addConvertingSetter(m, propertyName, converter, type); | |||||
| addConvertingSetter(m, depth, propertyName, converter); | |||||
| return; | return; | ||||
| } | } | ||||
| if (type.equals(String.class)) { | if (type.equals(String.class)) { | ||||
| attributeSetters.put(propertyName.toLowerCase(), | |||||
| new AttributeSetter() { | |||||
| public void set(Object parent, String value) | |||||
| throws InvocationTargetException, | |||||
| IllegalAccessException { | |||||
| m.invoke(parent, new String[]{value}); | |||||
| } | |||||
| }); | |||||
| addAttributeSetter(propertyName, new AttributeSetter(m, depth)); | |||||
| return; | return; | ||||
| } | } | ||||
| try { | try { | ||||
| final Constructor c = | final Constructor c = | ||||
| type.getConstructor(new Class[]{java.lang.String.class}); | type.getConstructor(new Class[]{java.lang.String.class}); | ||||
| attributeSetters.put(propertyName.toLowerCase(), | |||||
| new AttributeSetter() { | |||||
| public void set(Object parent, String value) | |||||
| throws InvocationTargetException, | |||||
| IllegalAccessException, ExecutionException { | |||||
| try { | |||||
| Object newValue | |||||
| = c.newInstance(new String[]{value}); | |||||
| m.invoke(parent, new Object[]{newValue}); | |||||
| } catch (InstantiationException ie) { | |||||
| throw new ExecutionException(ie); | |||||
| } | |||||
| } | |||||
| }); | |||||
| addAttributeSetter(propertyName, new AttributeSetter(m, depth, c)); | |||||
| return; | return; | ||||
| } catch (NoSuchMethodException nme) { | } catch (NoSuchMethodException nme) { | ||||
| // ignore | // ignore | ||||
| @@ -435,7 +422,7 @@ public class Reflector implements Setter { | |||||
| Converter converter | Converter converter | ||||
| = (Converter)converters.get(converterType); | = (Converter)converters.get(converterType); | ||||
| if (converter.canConvertSubType(type)) { | if (converter.canConvertSubType(type)) { | ||||
| addConvertingSetter(m, propertyName, converter, type); | |||||
| addConvertingSetter(m, depth, propertyName, converter); | |||||
| return; | return; | ||||
| } | } | ||||
| } | } | ||||
| @@ -484,23 +471,16 @@ public class Reflector implements Setter { | |||||
| * Add an attribute setter with an associated converter | * Add an attribute setter with an associated converter | ||||
| * | * | ||||
| * @param m the attribute setter method | * @param m the attribute setter method | ||||
| * @param depth the depth of this method's declaration in the class | |||||
| * hierarchy | |||||
| * @param propertyName the name of the attribute this method supports | * @param propertyName the name of the attribute this method supports | ||||
| * @param converter the converter to be used to construct the value | * @param converter the converter to be used to construct the value | ||||
| * expected by the method. | * expected by the method. | ||||
| * @param type the type expected by the method. | |||||
| */ | */ | ||||
| private void addConvertingSetter(final Method m, String propertyName, | |||||
| final Converter converter, | |||||
| final Class type) { | |||||
| attributeSetters.put(propertyName.toLowerCase(), | |||||
| new AttributeSetter() { | |||||
| public void set(Object obj, String value) | |||||
| throws InvocationTargetException, ExecutionException, | |||||
| IllegalAccessException { | |||||
| Object convertedValue = converter.convert(value, type); | |||||
| m.invoke(obj, new Object[]{convertedValue}); | |||||
| } | |||||
| }); | |||||
| private void addConvertingSetter(Method m, int depth, | |||||
| String propertyName, Converter converter) { | |||||
| addAttributeSetter(propertyName, | |||||
| new AttributeSetter(m, depth, converter)); | |||||
| } | } | ||||
| } | } | ||||
| @@ -885,8 +885,10 @@ public class Project implements org.apache.ant.common.event.BuildListener { | |||||
| * | * | ||||
| * @param taskType the name of the task to be created. | * @param taskType the name of the task to be created. | ||||
| * @return the created task instance | * @return the created task instance | ||||
| * | |||||
| * @exception BuildException if there is a build problem | |||||
| */ | */ | ||||
| public Task createTask(String taskType) { | |||||
| public Task createTask(String taskType) throws BuildException { | |||||
| Task task = null; | Task task = null; | ||||
| Class taskClass = (Class)taskClassDefinitions.get(taskType); | Class taskClass = (Class)taskClassDefinitions.get(taskType); | ||||