org.fusesource.hawtdispatch.internal.util.IntrospectionSupport Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hawtdispatch Show documentation
Show all versions of hawtdispatch Show documentation
HawtDispatch: The libdispatch style API for Java
/**
* 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.fusesource.hawtdispatch.internal.util;
import java.beans.PropertyEditor;
import java.beans.PropertyEditorManager;
import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
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;
public final class IntrospectionSupport {
static {
// Add Spring and ActiveMQ specific property editors
String[] additionalPath = new String[] {
"org.springframework.beans.propertyeditors",
"org.apache.activemq.util" };
String[] searchPath = (String[]) Array.newInstance(String.class,
PropertyEditorManager.getEditorSearchPath().length
+ additionalPath.length);
System.arraycopy(PropertyEditorManager.getEditorSearchPath(), 0,
searchPath, 0,
PropertyEditorManager.getEditorSearchPath().length);
System.arraycopy(additionalPath, 0, searchPath, PropertyEditorManager
.getEditorSearchPath().length, additionalPath.length);
PropertyEditorManager.setEditorSearchPath(searchPath);
}
private IntrospectionSupport() {
}
public static boolean getProperties(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.");
}
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("is") || 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;
}
if (name.startsWith("get")) {
name = name.substring(3, 4).toLowerCase()
+ name.substring(4);
} else {
name = name.substring(2, 3).toLowerCase()
+ name.substring(3);
}
props.put(optionPrefix + name, strValue);
rc = true;
} catch (Throwable ignore) {
}
}
}
return rc;
}
public static 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 = 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 rc = new HashMap(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;
}
public static Class> getPropertyType(Object target, String name) {
Class> clazz = target.getClass();
Method setter = findSetterMethod(clazz, name);
if (setter == null) {
return null;
}
return setter.getParameterTypes()[0];
}
public 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) {
if( type.isArray() ) {
if( value.getClass().isArray() ) {
int length = Array.getLength(value);
Class> componentType = type.getComponentType();
Object rc = Array.newInstance(componentType, length);
for (int i = 0; i < length; i++) {
Object o = Array.get(value, i);
Array.set(rc, i, convert(o, componentType));
}
return rc;
}
}
PropertyEditor editor = PropertyEditorManager.findEditor(type);
if (editor != null) {
editor.setAsText(value.toString());
return editor.getValue();
}
return null;
}
public static String convertToString(Object value, Class> type) {
PropertyEditor editor = PropertyEditorManager.findEditor(type);
if (editor != null) {
editor.setValue(value);
return editor.getAsText();
}
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 ) {
return method;
}
}
return null;
}
private static boolean isSettableType(Class> clazz) {
if (PropertyEditorManager.findEditor(clazz) != null) {
return true;
}
return false;
}
public static String toString(Object target) {
return toString(target, Object.class, null, null);
}
public static String toString(Object target, String...fields) {
return toString(target, Object.class, null, fields);
}
public static String toString(Object target, Class> stopClass) {
return toString(target, stopClass, null, null);
}
public static String toString(Object target, Map overrideFields, String...fields) {
return toString(target, Object.class, overrideFields, fields);
}
public static String toString(Object target, Class> stopClass, Map overrideFields, String fields[]) {
try {
LinkedHashMap map = new LinkedHashMap();
addFields(target, target.getClass(), stopClass, map);
if (overrideFields != null) {
for(String key : overrideFields.keySet()) {
Object value = overrideFields.get(key);
map.put(key, value);
}
}
if( fields!=null ) {
map.keySet().retainAll(Arrays.asList(fields));
}
boolean useMultiLine=false;
LinkedHashMap props = new LinkedHashMap();
for (Entry entry : map.entrySet()) {
String key = entry.getKey();
String value = null;
if( entry.getValue() !=null ) {
value = entry.getValue().toString();
if( value!=null && ( value.indexOf('\n')>=0 || (key.length()+value.length())>70 ) ) {
useMultiLine=true;
}
}
props.put(key, value);
}
StringBuffer buffer = new StringBuffer();
if( useMultiLine) {
buffer.append("{\n");
boolean first = true;
for (Entry entry : props.entrySet()) {
if (first) {
first = false;
} else {
buffer.append(",\n");
}
buffer.append(" ");
buffer.append(entry.getKey());
buffer.append(": ");
buffer.append(StringSupport.indent(entry.getValue(), 2));
}
buffer.append("\n}");
} else {
buffer.append("{");
boolean first = true;
for (Entry entry : props.entrySet()) {
if (first) {
first = false;
} else {
buffer.append(", ");
}
buffer.append(entry.getKey());
buffer.append(": ");
String value = entry.getValue();
buffer.append(value);
}
buffer.append("}");
}
return buffer.toString();
} catch (Throwable e) {
e.printStackTrace();
return "Could not toString: "+e.toString();
}
}
public static String simpleName(Class> clazz) {
String name = clazz.getName();
int p = name.lastIndexOf(".");
if (p >= 0) {
name = name.substring(p + 1);
}
return name;
}
private static void addFields(Object target, Class> startClass, Class> stopClass, LinkedHashMap 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())) {
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();
}
}
}
}