All Downloads are FREE. Search and download functionalities are using the official Maven repository.

bitronix.tm.utils.PropertyUtils Maven / Gradle / Ivy

There is a newer version: 62
Show newest version
/*
 * Copyright (C) 2006-2013 Bitronix Software (http://www.bitronix.be)
 *
 * Licensed 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 bitronix.tm.utils;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.TreeMap;

/**
 * Smart reflection helper.
 *
 * @author Ludovic Orban
 */
public final class PropertyUtils
{
	private static final String DASH_QUOTE_SPACE = "' - '";
	private static final String IS_NOT_ACCESSIBLE = "' is not accessible";
	private static final String PROPERTY_STRING = "property '";
	private static final String ACCESS_EXCEPTION = "' access threw an exception";
	private static final String IN_CLASS_STRING = "' in class '";

	/**
	 * Constructor PropertyUtils creates a new PropertyUtils instance.
	 */
	private PropertyUtils()
	{
		//No config required
	}

	/**
	 * Set a {@link Map} of direct or indirect properties on the target object.
	 *
	 * @param target
	 * 		the target object on which to set the properties.
	 * @param properties
	 * 		a {@link Map} of String/Object pairs.
	 *
	 * @throws PropertyException
	 * 		if an error happened while trying to set a property.
	 */
	public static void setProperties(Object target, Map properties)
	{
		for (Map.Entry entry : properties.entrySet())
		{
			String name = entry.getKey();
			Object value = entry.getValue();
			setProperty(target, name, value);
		}
	}

	/**
	 * Set a direct or indirect property (dotted property: prop1.prop2.prop3) on the target object. This method tries
	 * to be smart in the way that intermediate properties currently set to null are set if it is possible to create
	 * and set an object. Conversions from propertyValue to the proper destination type are performed when possible.
	 *
	 * @param target
	 * 		the target object on which to set the property.
	 * @param propertyName
	 * 		the name of the property to set.
	 * @param propertyValue
	 * 		the value of the property to set.
	 *
	 * @throws PropertyException
	 * 		if an error happened while trying to set the property.
	 */
	public static void setProperty(Object target, String propertyName, Object propertyValue)
	{
		String[] propertyNames = propertyName.split("\\.");

		StringBuilder visitedPropertyName = new StringBuilder();
		Object currentTarget = target;
		Object parentTarget = target;
		String parentName = null;
		int i = 0;
		while (i < propertyNames.length - 1)
		{
			String name = propertyNames[i];
			Object result = callGetter(currentTarget, name);
			if (result == null)
			{
				// try to instantiate the object & set it in place
				Class propertyType = getPropertyType(target, name);
				try
				{
					result = propertyType.getDeclaredConstructor()
					                     .newInstance();
				}
				catch (InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException ex)
				{
					throw new PropertyException("cannot set property '" + propertyName + DASH_QUOTE_SPACE + name + "' is null and cannot be auto-filled", ex);
				}
				callSetter(currentTarget, name, result);
			}

			parentTarget = currentTarget;
			currentTarget = result;
			visitedPropertyName.append(name);
			visitedPropertyName.append('.');
			i++;

			// if it's a Map object -> the non-visited part of the key should be used
			// as this Map's object key so stop iterating over the dotted properties.
			if (currentTarget instanceof Map)
			{
				parentName = name;
				break;
			}
		}

		String lastPropertyName = propertyName.substring(visitedPropertyName.length());
		if (currentTarget instanceof Map)
		{
			@SuppressWarnings("unchecked")
			Map p = (Map) currentTarget;
			p.put(lastPropertyName, propertyValue.toString());
			if (parentName != null)
			{
				callSetter(parentTarget, parentName, p);
			}
		}
		else
		{
			setDirectProperty(currentTarget, lastPropertyName, propertyValue);
		}
	}

	/**
	 * Method callGetter ...
	 *
	 * @param target
	 * 		of type Object
	 * @param propertyName
	 * 		of type String
	 *
	 * @return Object
	 *
	 * @throws PropertyException
	 * 		when
	 */
	private static Object callGetter(Object target, String propertyName)
	{
		Method getter = getGetter(target, propertyName);
		try
		{
			return getter.invoke(target);
		}
		catch (IllegalAccessException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + IS_NOT_ACCESSIBLE, ex);
		}
		catch (InvocationTargetException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + ACCESS_EXCEPTION, ex);
		}
	}

	/**
	 * Method getPropertyType ...
	 *
	 * @param target
	 * 		of type Object
	 * @param propertyName
	 * 		of type String
	 *
	 * @return Class
	 */
	private static Class getPropertyType(Object target, String propertyName)
	{
		String getterName = "get" + propertyName.substring(0, 1)
		                                        .toUpperCase() + propertyName.substring(1);
		String getterIsName = "is" + propertyName.substring(0, 1)
		                                         .toUpperCase() + propertyName.substring(1);
		Method[] methods = target.getClass()
		                         .getMethods();
		for (Method method : methods)
		{
			if ((method.getName()
			           .equals(getterName) || method.getName()
			                                        .equals(getterIsName)) && !method.getReturnType()
			                                                                         .equals(void.class) && method.getParameterTypes().length == 0)
			{
				return method.getReturnType();
			}
		}
		throw new PropertyException("no property '" + propertyName + IN_CLASS_STRING + target.getClass()
		                                                                                     .getName() + "'");
	}

	/**
	 * Method callSetter ...
	 *
	 * @param target
	 * 		of type Object
	 * @param propertyName
	 * 		of type String
	 * @param parameter
	 * 		of type Object
	 *
	 * @throws PropertyException
	 * 		when
	 */
	private static void callSetter(Object target, String propertyName, Object parameter)
	{
		Method setter = getSetter(target, propertyName);
		try
		{
			setter.invoke(target, parameter);
		}
		catch (IllegalAccessException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + IS_NOT_ACCESSIBLE, ex);
		}
		catch (InvocationTargetException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + ACCESS_EXCEPTION, ex);
		}
	}

	/**
	 * Set a direct property on the target object. Conversions from propertyValue to the proper destination type
	 * are performed whenever possible.
	 *
	 * @param target
	 * 		the target object on which to set the property.
	 * @param propertyName
	 * 		the name of the property to set.
	 * @param propertyValue
	 * 		the value of the property to set.
	 *
	 * @throws PropertyException
	 * 		if an error happened while trying to set the property.
	 */
	private static void setDirectProperty(Object target, String propertyName, Object propertyValue)
	{
		Method setter = getSetter(target, propertyName);
		Class parameterType = setter.getParameterTypes()[0];
		try
		{
			if (propertyValue != null)
			{
				Object transformedPropertyValue = transform(propertyValue, parameterType);
				setter.invoke(target, transformedPropertyValue);
			}
			else
			{
				setter.invoke(target);
			}
		}
		catch (IllegalAccessException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + IS_NOT_ACCESSIBLE, ex);
		}
		catch (InvocationTargetException ex)
		{
			throw new PropertyException(PROPERTY_STRING + propertyName + ACCESS_EXCEPTION, ex);
		}
	}

	/**
	 * Method getGetter ...
	 *
	 * @param target
	 * 		of type Object
	 * @param propertyName
	 * 		of type String
	 *
	 * @return Method
	 */
	private static Method getGetter(Object target, String propertyName)
	{
		String getterName = "get" + propertyName.substring(0, 1)
		                                        .toUpperCase() + propertyName.substring(1);
		String getterIsName = "is" + propertyName.substring(0, 1)
		                                         .toUpperCase() + propertyName.substring(1);
		Method[] methods = target.getClass()
		                         .getMethods();
		for (Method method : methods)
		{
			if ((method.getName()
			           .equals(getterName) || method.getName()
			                                        .equals(getterIsName)) && !method.getReturnType()
			                                                                         .equals(void.class) && method.getParameterTypes().length == 0)
			{
				return method;
			}
		}
		throw new PropertyException("no readable property '" + propertyName + IN_CLASS_STRING + target.getClass()
		                                                                                              .getName() + "'");
	}

	/**
	 * Method getSetter ...
	 *
	 * @param target
	 * 		of type Object
	 * @param propertyName
	 * 		of type String
	 *
	 * @return Method
	 */
	private static Method getSetter(Object target, String propertyName)
	{
		if (propertyName == null || "".equals(propertyName))
		{
			throw new PropertyException("encountered invalid null or empty property name");
		}
		String setterName = "set" + propertyName.substring(0, 1)
		                                        .toUpperCase() + propertyName.substring(1);
		Method[] methods = target.getClass()
		                         .getMethods();
		for (Method method : methods)
		{
			if (method.getName()
			          .equals(setterName) && method.getReturnType()
			                                       .equals(void.class) && method.getParameterTypes().length == 1)
			{
				return method;
			}
		}
		throw new PropertyException("no writeable property '" + propertyName + IN_CLASS_STRING + target.getClass()
		                                                                                               .getName() + "'");
	}

	/**
	 * Method transform ...
	 *
	 * @param value
	 * 		of type Object
	 * @param destinationClass
	 * 		of type Class
	 *
	 * @return Object
	 */
	private static Object transform(Object value, Class destinationClass)
	{
		if (value.getClass() == destinationClass)
		{
			return value;
		}

		if (value.getClass() == boolean.class || value.getClass() == Boolean.class ||
		    value.getClass() == byte.class || value.getClass() == Byte.class ||
		    value.getClass() == short.class || value.getClass() == Short.class ||
		    value.getClass() == int.class || value.getClass() == Integer.class ||
		    value.getClass() == long.class || value.getClass() == Long.class ||
		    value.getClass() == float.class || value.getClass() == Float.class ||
		    value.getClass() == double.class || value.getClass() == Double.class
		)
		{
			return value;
		}

		if ((destinationClass == boolean.class || destinationClass == Boolean.class) && value.getClass() == String.class)
		{
			return Boolean.valueOf((String) value);
		}
		if ((destinationClass == byte.class || destinationClass == Byte.class) && value.getClass() == String.class)
		{
			return Byte.valueOf((String) value);
		}
		if ((destinationClass == short.class || destinationClass == Short.class) && value.getClass() == String.class)
		{
			return Short.valueOf((String) value);
		}
		if ((destinationClass == int.class || destinationClass == Integer.class) && value.getClass() == String.class)
		{
			return Integer.valueOf((String) value);
		}
		if ((destinationClass == long.class || destinationClass == Long.class) && value.getClass() == String.class)
		{
			return Long.valueOf((String) value);
		}
		if ((destinationClass == float.class || destinationClass == Float.class) && value.getClass() == String.class)
		{
			return Float.valueOf((String) value);
		}
		if ((destinationClass == double.class || destinationClass == Double.class) && value.getClass() == String.class)
		{
			return Double.valueOf((String) value);
		}

		throw new PropertyException("cannot convert values of type '" + value.getClass()
		                                                                     .getName() + "' into type '" + destinationClass + "'");
	}

	/**
	 * Return a comma-separated String of r/w properties of the specified object.
	 *
	 * @param obj
	 * 		the object to introspect.
	 *
	 * @return a a comma-separated String of r/w properties.
	 */
	public static String propertiesToString(Object obj)
	{
		StringBuilder sb = new StringBuilder();
		Map properties = new TreeMap<>(getProperties(obj));
		Iterator it = properties.keySet()
		                                .iterator();
		while (it.hasNext())
		{
			String property = it.next();
			Object val = getProperty(obj, property);
			sb.append(property);
			sb.append("=");
			sb.append(val);
			if (it.hasNext())
			{
				sb.append(", ");
			}
		}
		return sb.toString();
	}

	/**
	 * Build a map of direct javabeans properties of the target object. Only read/write properties (ie: those who have
	 * both a getter and a setter) are returned.
	 *
	 * @param target
	 * 		the target object from which to get properties names.
	 *
	 * @return a Map of String with properties names as key and their values
	 *
	 * @throws PropertyException
	 * 		if an error happened while trying to get a property.
	 */
	@SuppressWarnings("WeakerAccess")
	public static Map getProperties(Object target)
	{
		Map properties = new HashMap<>();
		Class clazz = target.getClass();
		Method[] methods = clazz.getMethods();
		for (Method method : methods)
		{
			String name = method.getName();
			if (method.getModifiers() == Modifier.PUBLIC && method.getParameterTypes().length == 0 && (name.startsWith("get") || name.startsWith("is")) &&
			    containsSetterForGetter(clazz, method))
			{
				String propertyName;
				if (name.startsWith("get"))
				{
					propertyName = Character.toLowerCase(name.charAt(3)) + name.substring(4);
				}
				else if (name.startsWith("is"))
				{
					propertyName = Character.toLowerCase(name.charAt(2)) + name.substring(3);
				}
				else
				{
					throw new PropertyException("method '" + name + "' is not a getter, thereof no setter can be found");
				}

				try
				{
					Object propertyValue = method.invoke(target);
					if (propertyValue instanceof Map)
					{
						@SuppressWarnings("unchecked")
						Map propertiesContent = getNestedProperties(propertyName, (Map) propertyValue);
						properties.putAll(propertiesContent);
					}
					else
					{
						properties.put(propertyName, propertyValue);
					}
				}
				catch (IllegalAccessException | InvocationTargetException ex)
				{
					throw new PropertyException("cannot set property '" + propertyName + DASH_QUOTE_SPACE + name + "' is null and cannot be auto-filled", ex);
				}

			} // if
		} // for
		return properties;
	}

	/**
	 * Get a direct or indirect property (dotted property: prop1.prop2.prop3) on the target object.
	 *
	 * @param target
	 * 		the target object from which to get the property.
	 * @param propertyName
	 * 		the name of the property to get.
	 *
	 * @return the value of the specified property.
	 *
	 * @throws PropertyException
	 * 		if an error happened while trying to get the property.
	 */
	public static Object getProperty(Object target, String propertyName)
	{
		String[] propertyNames = propertyName.split("\\.");
		Object currentTarget = target;
		for (int i = 0; i < propertyNames.length; i++)
		{
			String name = propertyNames[i];
			Object result = callGetter(currentTarget, name);
			if (result == null && i < propertyNames.length - 1)
			{
				throw new PropertyException("cannot get property '" + propertyName + DASH_QUOTE_SPACE + name + "' is null");
			}
			currentTarget = result;
		}

		return currentTarget;
	}

	/**
	 * Method containsSetterForGetter ...
	 *
	 * @param clazz
	 * 		of type Class ?
	 * @param method
	 * 		of type Method
	 *
	 * @return boolean
	 */
	private static boolean containsSetterForGetter(Class clazz, Method method)
	{
		String methodName = method.getName();
		String setterName;

		if (methodName.startsWith("get"))
		{
			setterName = "set" + methodName.substring(3);
		}
		else if (methodName.startsWith("is"))
		{
			setterName = "set" + methodName.substring(2);
		}
		else
		{
			throw new PropertyException("method '" + methodName + "' is not a getter, thereof no setter can be found");
		}

		Method[] methods = clazz.getMethods();
		for (Method m : methods)
		{
			if (m.getName()
			     .equals(setterName))
			{
				return true;
			}
		}
		return false;
	}

	/**
	 * Method getNestedProperties ...
	 *
	 * @param prefix
	 * 		of type String
	 * @param properties
	 * 		of type Map String, Object
	 *
	 * @return Map String,Object
	 */
	private static Map getNestedProperties(String prefix, Map properties)
	{
		Map result = new HashMap<>();
		for (Map.Entry entry : properties.entrySet())
		{
			String name = entry.getKey();
			String value = (String) entry.getValue();
			result.put(prefix + '.' + name, value);
		}
		return result;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy