conor 02/03/24 21:40:34
Modified: proposal/mutant build.xml
proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution
ClassIntrospector.java Reflector.java
proposal/mutant/src/java/antlibs/ant1compat/org/apache/tools/ant
Project.java
Added: proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution
AttributeSetter.java
Log:
MOve Attribute Setter outof Reflector to its own class
Revision Changes Path
1.16 +2 -1 jakarta-ant/proposal/mutant/build.xml
Index: build.xml
===================================================================
RCS file: /home/cvs/jakarta-ant/proposal/mutant/build.xml,v
retrieving revision 1.15
retrieving revision 1.16
diff -u -w -u -r1.15 -r1.16
--- build.xml 6 Mar 2002 08:17:06 -0000 1.15
+++ build.xml 25 Mar 2002 05:40:33 -0000 1.16
@@ -201,7 +201,8 @@
useexternalfile="yes"
sourcepath="${java.dir}/antcore:${java.dir}/init:${java.dir}/common:${java.dir}/cli:${java.dir}/start"
destdir="${javadocs.dir}"
- author="true" private ="true"
+ author="true"
+ private ="true"
version="true"
windowtitle="Mutant API"
doctitle="Mutant">
1.6 +26 -2
jakarta-ant/proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution/ClassIntrospector.java
Index: ClassIntrospector.java
===================================================================
RCS file:
/home/cvs/jakarta-ant/proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution/ClassIntrospector.java,v
retrieving revision 1.5
retrieving revision 1.6
diff -u -w -u -r1.5 -r1.6
--- ClassIntrospector.java 18 Mar 2002 02:44:24 -0000 1.5
+++ ClassIntrospector.java 25 Mar 2002 05:40:33 -0000 1.6
@@ -54,6 +54,7 @@
package org.apache.ant.antcore.execution;
import java.lang.reflect.Method;
import java.util.Map;
+import java.util.HashMap;
/**
* Introspects a class and builds a reflector for setting values on
@@ -67,6 +68,27 @@
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
*
* @param bean the class which is introspected
@@ -75,6 +97,7 @@
*/
public ClassIntrospector(final Class bean, Map converters) {
reflector = new Reflector();
+ getDepths(bean);
Method[] methods = bean.getMethods();
for (int i = 0; i < methods.length; i++) {
@@ -93,8 +116,9 @@
&& returnType.equals(Void.TYPE)
&& args.length == 1
&& !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")
&& name.length() > 13
&& returnType.equals(Void.TYPE)
1.6 +45 -65
jakarta-ant/proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution/Reflector.java
Index: Reflector.java
===================================================================
RCS file:
/home/cvs/jakarta-ant/proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution/Reflector.java,v
retrieving revision 1.5
retrieving revision 1.6
diff -u -w -u -r1.5 -r1.6
--- Reflector.java 18 Mar 2002 02:44:24 -0000 1.5
+++ Reflector.java 25 Mar 2002 05:40:33 -0000 1.6
@@ -72,31 +72,6 @@
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
* object. The object being added will have been fully configured by Ant
* prior to calling this method.
@@ -360,6 +335,36 @@
}
/**
+ * 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
* particular nested element
*
@@ -375,51 +380,33 @@
* Add a method to the reflector for setting an attribute value
*
* @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 converters A map of converter classes used to convert strings
* 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)) {
// we have a converter to use to convert the String
// value into something the set method expects.
Converter converter = (Converter)converters.get(type);
- addConvertingSetter(m, propertyName, converter, type);
+ addConvertingSetter(m, depth, propertyName, converter);
return;
}
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;
}
try {
final Constructor c =
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;
} catch (NoSuchMethodException nme) {
// ignore
@@ -435,7 +422,7 @@
Converter converter
= (Converter)converters.get(converterType);
if (converter.canConvertSubType(type)) {
- addConvertingSetter(m, propertyName, converter,
type);
+ addConvertingSetter(m, depth, propertyName,
converter);
return;
}
}
@@ -484,23 +471,16 @@
* Add an attribute setter with an associated converter
*
* @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 converter the converter to be used to construct the value
* 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));
}
}
1.1
jakarta-ant/proposal/mutant/src/java/antcore/org/apache/ant/antcore/execution/AttributeSetter.java
Index: AttributeSetter.java
===================================================================
/*
* 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 [EMAIL PROTECTED]
*
* 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];
}
}
1.15 +3 -1
jakarta-ant/proposal/mutant/src/java/antlibs/ant1compat/org/apache/tools/ant/Project.java
Index: Project.java
===================================================================
RCS file:
/home/cvs/jakarta-ant/proposal/mutant/src/java/antlibs/ant1compat/org/apache/tools/ant/Project.java,v
retrieving revision 1.14
retrieving revision 1.15
diff -u -w -u -r1.14 -r1.15
--- Project.java 24 Mar 2002 13:28:31 -0000 1.14
+++ Project.java 25 Mar 2002 05:40:34 -0000 1.15
@@ -885,8 +885,10 @@
*
* @param taskType the name of the task to be created.
* @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;
Class taskClass = (Class)taskClassDefinitions.get(taskType);
--
To unsubscribe, e-mail: <mailto:[EMAIL PROTECTED]>
For additional commands, e-mail: <mailto:[EMAIL PROTECTED]>