/** * 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.activemq.util; 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.List; import java.util.Locale; import java.util.Map; import java.util.Map.Entry; import java.util.Set; import javax.net.ssl.SSLServerSocket; import org.apache.activemq.command.ActiveMQDestination; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public final class IntrospectionSupport { private static final Logger LOG = LoggerFactory.getLogger(IntrospectionSupport.class); 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 (Method method : methods) { String name = method.getName(); Class<?> type = method.getReturnType(); Class<?> params[] = method.getParameterTypes(); if ((name.startsWith("is") || name.startsWith("get")) && params.length == 0 && type != null) { try { Object value = method.invoke(target); if (value == null) { continue; } String strValue = convertToString(value, type); if (strValue == null) { continue; } if (name.startsWith("get")) { name = name.substring(3, 4).toLowerCase(Locale.ENGLISH) + name.substring(4); } else { name = name.substring(2, 3).toLowerCase(Locale.ENGLISH) + name.substring(3); } props.put(optionPrefix + name, strValue); rc = true; } catch (Exception ignore) { } } } return rc; } public static boolean setProperties(Object target, Map<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."); } for (Iterator<String> 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<String, Object> 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) { return setProperties(target, props, true); } public static boolean setProperties(Object target, Map<?, ?> props, boolean removeIfSet) { 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())) { if (removeIfSet) { iter.remove(); } rc = true; } } return rc; } public static boolean setProperty(Object target, String name, Object value) { try { Class<?> clazz = target.getClass(); if (target instanceof SSLServerSocket) { // overcome illegal access issues with internal implementation class clazz = SSLServerSocket.class; } 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, value); } else { // We need to convert it setter.invoke(target, convert(value, setter.getParameterTypes()[0])); } return true; } catch (Exception e) { LOG.error(String.format("Could not set property %s on %s", name, target), e); return false; } } private static Object convert(Object value, Class to) { if (value == null) { // lets avoid NullPointerException when converting to boolean for null values if (boolean.class.isAssignableFrom(to)) { return Boolean.FALSE; } return null; } // eager same instance type test to avoid the overhead of invoking the type converter // if already same type if (to.isAssignableFrom(value.getClass())) { return to.cast(value); } // special for String[] as we do not want to use a PropertyEditor for that if (to.isAssignableFrom(String[].class)) { return StringArrayConverter.convertToStringArray(value); } // special for String to List<ActiveMQDestination> as we do not want to use a PropertyEditor for that if (value.getClass().equals(String.class) && to.equals(List.class)) { Object answer = StringToListOfActiveMQDestinationConverter.convertToActiveMQDestination(value); if (answer != null) { return answer; } } TypeConversionSupport.Converter converter = TypeConversionSupport.lookupConverter(value.getClass(), to); if (converter != null) { return converter.convert(value); } else { throw new IllegalArgumentException("Cannot convert from " + value.getClass() + " to " + to + " with value " + value); } } public static String convertToString(Object value, Class to) { if (value == null) { return null; } // already a String if (value instanceof String) { return (String) value; } // special for String[] as we do not want to use a PropertyEditor for that if (String[].class.isInstance(value)) { String[] array = (String[]) value; return StringArrayConverter.convertToString(array); } // special for String to List<ActiveMQDestination> as we do not want to use a PropertyEditor for that if (List.class.isInstance(value)) { // if the list is a ActiveMQDestination, then return a comma list String answer = StringToListOfActiveMQDestinationConverter.convertFromActiveMQDestination(value); if (answer != null) { return answer; } } TypeConversionSupport.Converter converter = TypeConversionSupport.lookupConverter(value.getClass(), String.class); if (converter != null) { return (String) converter.convert(value); } else { throw new IllegalArgumentException("Cannot convert from " + value.getClass() + " to " + to + " with value " + value); } } public static Method findSetterMethod(Class clazz, String name) { // Build the method name. name = "set" + Character.toUpperCase(name.charAt(0)) + name.substring(1); Method[] methods = clazz.getMethods(); for (Method method : methods) { Class<?> params[] = method.getParameterTypes(); if (method.getName().equals(name) && params.length == 1 ) { return method; } } return null; } public static Method findGetterMethod(Class clazz, String name) { // Build the method name. name = "get" + Character.toUpperCase(name.charAt(0)) + name.substring(1); Method[] methods = clazz.getMethods(); for (Method method : methods) { Class<?> params[] = method.getParameterTypes(); if (method.getName().equals(name) && params.length == 0 ) { return method; } } return null; } public static String toString(Object target) { return toString(target, Object.class, null); } public static String toString(Object target, Class stopClass) { return toString(target, stopClass, null); } public static String toString(Object target, Class stopClass, Map<String, Object> overrideFields) { LinkedHashMap<String, Object> map = new LinkedHashMap<String, Object>(); addFields(target, target.getClass(), stopClass, map); if (overrideFields != null) { for(String key : overrideFields.keySet()) { Object value = overrideFields.get(key); map.put(key, value); } } StringBuffer buffer = new StringBuffer(simpleName(target.getClass())); buffer.append(" {"); Set<Entry<String, Object>> entrySet = map.entrySet(); boolean first = true; for (Map.Entry<String,Object> entry : entrySet) { Object value = entry.getValue(); Object key = entry.getKey(); if (first) { first = false; } else { buffer.append(", "); } buffer.append(key); buffer.append(" = "); appendToString(buffer, key, value); } buffer.append("}"); return buffer.toString(); } protected static void appendToString(StringBuffer buffer, Object key, Object value) { if (value instanceof ActiveMQDestination) { ActiveMQDestination destination = (ActiveMQDestination)value; buffer.append(destination.getQualifiedName()); } else if (key.toString().toLowerCase(Locale.ENGLISH).contains("password")){ buffer.append("*****"); } else { buffer.append(value); } } 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<Object> stopClass, LinkedHashMap<String, Object> map) { if (startClass != stopClass) { addFields(target, startClass.getSuperclass(), stopClass, map); } Field[] fields = startClass.getDeclaredFields(); for (Field field : fields) { 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 (Exception e) { } } map.put(field.getName(), o); } catch (Exception e) { LOG.debug("Error getting field " + field + " on class " + startClass + ". This exception is ignored.", e); } } } }