blob: 063f78ccddf7cafb2fd9c7a820dded248ab18162 [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.myfaces.extensions.validator.util;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.StringTokenizer;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.apache.myfaces.extensions.validator.core.ExtValContext;
import org.apache.myfaces.extensions.validator.core.storage.PropertyStorage;
import org.apache.myfaces.extensions.validator.internal.UsageCategory;
import org.apache.myfaces.extensions.validator.internal.UsageInformation;
/**
* @author Gerhard Petracek
* @since 1.x.1
*/
@UsageInformation(UsageCategory.INTERNAL)
public class ReflectionUtils
{
private static final Logger LOGGER = Logger.getLogger(ReflectionUtils.class.getName());
public static Method tryToGetMethod(Class targetClass, String targetMethodName)
{
return tryToGetMethod(targetClass, targetMethodName, null);
}
public static Method tryToGetMethod(Class targetClass, String targetMethodName, Class... parameterTypes)
{
try
{
return getMethod(targetClass, targetMethodName, parameterTypes);
}
catch (Exception e)
{
//do nothing - it's just a try
return null;
}
}
public static Method getMethod(Class targetClass, String targetMethodName)
throws NoSuchMethodException
{
return getMethod(targetClass, targetMethodName, null);
}
public static Method getMethod(Class targetClass, String targetMethodName, Class... parameterTypes)
throws NoSuchMethodException
{
Class currentClass = targetClass;
Method targetMethod = null;
while (!Object.class.getName().equals(currentClass.getName()))
{
try
{
targetMethod = currentClass.getDeclaredMethod(targetMethodName, parameterTypes);
break;
}
catch (NoSuchMethodException e)
{
currentClass = currentClass.getSuperclass();
}
}
if(targetMethod == null)
{
for (Class currentInterface : targetClass.getInterfaces())
{
currentClass = currentInterface;
while (currentClass != null)
{
try
{
targetMethod = currentClass.getDeclaredMethod(targetMethodName, parameterTypes);
break;
}
catch (NoSuchMethodException e)
{
currentClass = currentClass.getSuperclass();
}
}
}
}
if(targetMethod != null)
{
return targetMethod;
}
throw new NoSuchMethodException("there is no method with the name '" + targetMethodName + "'" +
" class: " + targetClass.getName());
}
public static Object tryToInvokeMethod(Object target, Method method)
{
return tryToInvokeMethod(target, method, null);
}
public static Object tryToInvokeMethodOfClass(Class target, Method method)
{
return tryToInvokeMethodOfClass(target, method, null);
}
public static Object tryToInvokeMethodOfClass(Class target, Method method, Object[] args)
{
try
{
return invokeMethodOfClass(target, method, args);
}
catch (Exception e)
{
//do nothing - it's just a try
return null;
}
}
public static Object invokeMethodOfClass(Class target, Method method)
throws IllegalAccessException, InstantiationException, InvocationTargetException
{
return invokeMethod(target.newInstance(), method, null);
}
public static Object invokeMethodOfClass(Class target, Method method, Object... args)
throws IllegalAccessException, InstantiationException, InvocationTargetException
{
return invokeMethod(target.newInstance(), method, args);
}
public static Object tryToInvokeMethod(Object target, Method method, Object... args)
{
try
{
return invokeMethod(target, method, args);
}
catch (Exception e)
{
//do nothing - it's just a try
return null;
}
}
public static Object invokeMethod(Object target, Method method)
throws InvocationTargetException, IllegalAccessException
{
return invokeMethod(target, method, null);
}
public static Object invokeMethod(Object target, Method method, Object... args)
throws InvocationTargetException, IllegalAccessException
{
method.setAccessible(true);
return method.invoke(target, args);
}
public static Object getBaseOfPropertyChain(Object baseObject, String propertyChain)
{
StringTokenizer tokenizer = new StringTokenizer(propertyChain, ".");
Object currentBase = baseObject;
String currentProperty;
Method currentMethod;
while(tokenizer.hasMoreTokens())
{
currentProperty = tokenizer.nextToken();
//ignore the last property
if(!tokenizer.hasMoreTokens())
{
break;
}
//no is - it's only possible at properties not at bean level
currentMethod = tryToGetMethod(ProxyUtils.getUnproxiedClass(currentBase.getClass()),
"get" + currentProperty.substring(0, 1).toUpperCase() + currentProperty.substring(1));
currentBase = tryToInvokeMethod(currentBase, currentMethod);
}
return currentBase;
}
public static PropertyStorage getPropertyStorage()
{
return ExtValUtils.getStorage(PropertyStorage.class, PropertyStorage.class.getName());
}
@Deprecated
public static Method tryToGetMethodOfProperty(Class entity, String property)
{
return tryToGetMethodOfProperty(getPropertyStorage(), entity, property);
}
public static Method tryToGetMethodOfProperty(PropertyStorage storage, Class entity, String property)
{
if (isCachedMethod(storage, entity, property))
{
return getCachedMethod(storage, entity, property);
}
Method method = tryToGetReadMethod(entity, property);
tryToCacheMethod(storage, entity, property, method);
return method;
}
@Deprecated
public static Field tryToGetFieldOfProperty(Class entity, String property)
{
return tryToGetFieldOfProperty(getPropertyStorage(), entity, property);
}
public static Field tryToGetFieldOfProperty(PropertyStorage storage, Class entity, String property)
{
if (isCachedField(storage, entity, property))
{
return getCachedField(storage, entity, property);
}
Field field = null;
try
{
field = entity.getDeclaredField(property);
}
catch (Exception e)
{
try
{
try
{
field = entity.getDeclaredField("_" + property);
}
catch (Exception e1)
{
if (property.length() > 1
&& Character.isUpperCase(property.charAt(0))
&& Character.isUpperCase(property.charAt(1)))
{
//don't use Introspector#decapitalize here
field = entity.getDeclaredField(property.substring(0, 1).toLowerCase() + property.substring(1));
}
else
{
field = entity.getDeclaredField(Introspector.decapitalize(property));
}
}
}
catch (NoSuchFieldException e1)
{
LOGGER.log(Level.FINEST, "field " + property + " or _" + property + " not found", e1);
}
}
tryToCacheField(storage, entity, property, field);
return field;
}
private static void tryToCacheField(PropertyStorage storage, Class entity, String property, Field field)
{
if (!storage.containsField(entity, property))
{
storage.storeField(entity, property, field);
}
}
private static boolean isCachedField(PropertyStorage storage, Class entity, String property)
{
return storage.containsField(entity, property);
}
private static Method tryToGetReadMethod(Class baseBeanClass, String property)
{
Method method = ReflectionUtils.tryToGetReadMethodViaBeanInfo(baseBeanClass, property);
if (method == null)
{
method = ReflectionUtils.tryToGetReadMethodManually(baseBeanClass, property);
}
return method;
}
private static Method tryToGetReadMethodViaBeanInfo(Class entity, String property)
{
if (useBeanInfo())
{
try
{
BeanInfo beanInfo = Introspector.getBeanInfo(entity);
for (PropertyDescriptor propertyDescriptor : beanInfo
.getPropertyDescriptors())
{
if (property.equals(propertyDescriptor.getName())
&& propertyDescriptor.getReadMethod() != null)
{
return propertyDescriptor.getReadMethod();
}
}
}
catch (IntrospectionException e)
{
//do nothing
}
}
return null;
}
private static boolean useBeanInfo()
{
return Boolean.TRUE.equals(ExtValContext.getContext().getGlobalProperty(BeanInfo.class.getName()));
}
private static Method tryToGetReadMethodManually(Class entity, String property)
{
property = property.substring(0, 1).toUpperCase() + property.substring(1);
try
{
//changed to official bean spec. due to caching there is no performance issue any more
return entity.getDeclaredMethod("is" + property);
}
catch (NoSuchMethodException e)
{
try
{
return entity.getDeclaredMethod("get" + property);
}
catch (NoSuchMethodException e1)
{
LOGGER.finest("method not found - class: " + entity.getName()
+ " - methods: " + "get" + property + " " + "is" + property);
return null;
}
}
}
private static Field getCachedField(PropertyStorage storage, Class entity, String property)
{
return storage.getField(entity, property);
}
private static boolean isCachedMethod(PropertyStorage storage, Class entity, String property)
{
return storage.containsMethod(entity, property);
}
private static void tryToCacheMethod(PropertyStorage storage, Class entity, String property, Method method)
{
if (!storage.containsMethod(entity, property))
{
storage.storeMethod(entity, property, method);
}
}
private static Method getCachedMethod(PropertyStorage storage, Class entity, String property)
{
return storage.getMethod(entity, property);
}
}