blob: 524d54a221d0f3b2e066db853d062fed73b48eb4 [file] [log] [blame]
/*
* 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.
*/
package org.apache.openejb.util;
import java.beans.Introspector;
import java.beans.PropertyEditor;
import java.beans.PropertyEditorManager;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
public class IntrospectionSupport {
static public boolean getProperties(Object target, Map<String, String> props,
String optionPrefix) {
boolean rc = false;
if (target == null)
throw new IllegalArgumentException("target was null.");
if (props == null)
throw new IllegalArgumentException("props was null.");
if (optionPrefix == null)
optionPrefix = "";
Class clazz = target.getClass();
Method[] methods = clazz.getMethods();
for (int i = 0; i < methods.length; i++) {
Method method = methods[i];
String name = method.getName();
Class type = method.getReturnType();
Class params[] = method.getParameterTypes();
if (name.startsWith("get") && params.length == 0 && type != null
&& isSettableType(type)) {
try {
Object value = method.invoke(target, new Object[] {});
if (value == null)
continue;
String strValue = convertToString(value, type);
if (strValue == null)
continue;
name = name.substring(3, 4).toLowerCase()
+ name.substring(4);
props.put(optionPrefix + name, strValue);
rc = true;
} catch (Throwable ignore) {
}
}
}
return rc;
}
static public boolean setProperties(Object target, Map props,
String optionPrefix) {
boolean rc = false;
if (target == null)
throw new IllegalArgumentException("target was null.");
if (props == null)
throw new IllegalArgumentException("props was null.");
for (Iterator iter = props.keySet().iterator(); iter.hasNext();) {
String name = (String) iter.next();
if (name.startsWith(optionPrefix)) {
Object value = props.get(name);
name = name.substring(optionPrefix.length());
if (setProperty(target, name, value)) {
iter.remove();
rc = true;
}
}
}
return rc;
}
public static Map extractProperties(Map props, String optionPrefix) {
if (props == null)
throw new IllegalArgumentException("props was null.");
HashMap<String, Object> rc = new HashMap<String, Object>(props.size());
for (Iterator iter = props.keySet().iterator(); iter.hasNext();) {
String name = (String) iter.next();
if (name.startsWith(optionPrefix)) {
Object value = props.get(name);
name = name.substring(optionPrefix.length());
rc.put(name, value);
iter.remove();
}
}
return rc;
}
public static boolean setProperties(Object target, Map props) {
boolean rc = false;
if (target == null)
throw new IllegalArgumentException("target was null.");
if (props == null)
throw new IllegalArgumentException("props was null.");
for (Iterator iter = props.entrySet().iterator(); iter.hasNext();) {
Map.Entry entry = (Entry) iter.next();
if (setProperty(target, (String) entry.getKey(), entry.getValue())) {
iter.remove();
rc = true;
}
}
return rc;
}
private static boolean setProperty(Object target, String name, Object value) {
try {
Class clazz = target.getClass();
Method setter = findSetterMethod(clazz, name);
if (setter == null)
return false;
// If the type is null or it matches the needed type, just use the
// value directly
if (value == null
|| value.getClass() == setter.getParameterTypes()[0]) {
setter.invoke(target, new Object[] { value });
} else {
// We need to convert it
setter.invoke(target, new Object[] { convert(value, setter
.getParameterTypes()[0]) });
}
return true;
} catch (Throwable ignore) {
return false;
}
}
private static Object convert(Object value, Class type)
throws URISyntaxException {
PropertyEditor editor = PropertyEditorManager.findEditor(type);
if (editor != null) {
editor.setAsText(value.toString());
return editor.getValue();
}
if (type == URI.class) {
return URLs.uri(value.toString());
}
return null;
}
private static String convertToString(Object value, Class type)
throws URISyntaxException {
PropertyEditor editor = PropertyEditorManager.findEditor(type);
if (editor != null) {
editor.setValue(value);
return editor.getAsText();
}
if (type == URI.class) {
return ((URI) value).toString();
}
return null;
}
private static Method findSetterMethod(Class clazz, String name) {
// Build the method name.
name = "set" + name.substring(0, 1).toUpperCase() + name.substring(1);
Method[] methods = clazz.getMethods();
for (int i = 0; i < methods.length; i++) {
Method method = methods[i];
Class params[] = method.getParameterTypes();
if (method.getName().equals(name) && params.length == 1
&& isSettableType(params[0])) {
return method;
}
}
return null;
}
private static boolean isSettableType(Class clazz) {
if (PropertyEditorManager.findEditor(clazz) != null)
return true;
if (clazz == URI.class)
return true;
if (clazz == Boolean.class)
return true;
return false;
}
static public String toString(Object target) {
return toString(target, Object.class);
}
static public String toString(Object target, Class stopClass) {
LinkedHashMap<String, Object> map = new LinkedHashMap<String, Object>();
addFields(target, target.getClass(), stopClass, map);
StringBuilder buffer = new StringBuilder(simpleName(target.getClass()));
buffer.append(" {");
Set entrySet = map.entrySet();
boolean first = true;
for (Iterator iter = entrySet.iterator(); iter.hasNext();) {
Map.Entry entry = (Map.Entry) iter.next();
if (first) {
first = false;
} else {
buffer.append(", ");
}
buffer.append(entry.getKey());
buffer.append(" = ");
appendToString(buffer, entry.getValue());
}
buffer.append("}");
return buffer.toString();
}
protected static void appendToString(StringBuilder buffer, Object value) {
buffer.append(value);
}
static public String simpleName(Class clazz) {
String name = clazz.getName();
int p = name.lastIndexOf(".");
if (p >= 0) {
name = name.substring(p + 1);
}
return name;
}
static private void addFields(Object target, Class startClass,
Class stopClass, LinkedHashMap<String, Object> map) {
if (startClass != stopClass)
addFields(target, startClass.getSuperclass(), stopClass, map);
Field[] fields = startClass.getDeclaredFields();
for (int i = 0; i < fields.length; i++) {
Field field = fields[i];
if (Modifier.isStatic(field.getModifiers())
|| Modifier.isTransient(field.getModifiers())
|| Modifier.isPrivate(field.getModifiers())) {
continue;
}
try {
field.setAccessible(true);
Object o = field.get(target);
if (o != null && o.getClass().isArray()) {
try {
o = Arrays.asList((Object[]) o);
} catch (Throwable e) {
}
}
map.put(field.getName(), o);
} catch (Throwable e) {
e.printStackTrace();
}
}
}
public static Class getPropertyType(Class clazz, String propertyName) throws NoSuchFieldException {
do {
try {
return clazz.getDeclaredField(propertyName).getType();
} catch (NoSuchFieldException e) {
//look at superclass
}
for (Method method: clazz.getDeclaredMethods()) {
if (method.getReturnType() == void.class && method.getParameterTypes().length == 1) {
String methodName = method.getName();
if (methodName.startsWith("set")) {
String type = Introspector.decapitalize(methodName.substring(3));
if (propertyName.equals(type)) {
return method.getParameterTypes()[0];
}
}
}
}
clazz = clazz.getSuperclass();
} while (clazz != null);
throw new NoSuchFieldException(propertyName);
}
}