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

org.apache.commons.configuration.beanutils.BeanHelper Maven / Gradle / Ivy

There is a newer version: 4.15.102
Show newest version
/*
 * 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.commons.configuration.beanutils;

import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;

import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.PropertyUtils;
import org.apache.commons.configuration.ConfigurationRuntimeException;
import org.apache.commons.lang.ClassUtils;

/**
 * 

* A helper class for creating bean instances that are defined in configuration * files. *

*

* This class provides static utility methods related to bean creation * operations. These methods simplify such operations because a client need not * deal with all involved interfaces. Usually, if a bean declaration has already * been obtained, a single method call is necessary to create a new bean * instance. *

*

* This class also supports the registration of custom bean factories. * Implementations of the {@link BeanFactory} interface can be * registered under a symbolic name using the {@code registerBeanFactory()} * method. In the configuration file the name of the bean factory can be * specified in the bean declaration. Then this factory will be used to create * the bean. *

* * @since 1.3 * @author Commons * Configuration team * @version $Id: BeanHelper.java 1534393 2013-10-21 22:02:27Z henning $ */ public final class BeanHelper { /** Stores a map with the registered bean factories. */ private static final Map BEAN_FACTORIES = Collections .synchronizedMap(new HashMap()); /** * Stores the default bean factory, which will be used if no other factory * is provided. */ private static BeanFactory defaultBeanFactory = DefaultBeanFactory.INSTANCE; /** * Private constructor, so no instances can be created. */ private BeanHelper() { } /** * Register a bean factory under a symbolic name. This factory object can * then be specified in bean declarations with the effect that this factory * will be used to obtain an instance for the corresponding bean * declaration. * * @param name the name of the factory * @param factory the factory to be registered */ public static void registerBeanFactory(String name, BeanFactory factory) { if (name == null) { throw new IllegalArgumentException( "Name for bean factory must not be null!"); } if (factory == null) { throw new IllegalArgumentException("Bean factory must not be null!"); } BEAN_FACTORIES.put(name, factory); } /** * Deregisters the bean factory with the given name. After that this factory * cannot be used any longer. * * @param name the name of the factory to be deregistered * @return the factory that was registered under this name; null if * there was no such factory */ public static BeanFactory deregisterBeanFactory(String name) { return BEAN_FACTORIES.remove(name); } /** * Returns a set with the names of all currently registered bean factories. * * @return a set with the names of the registered bean factories */ public static Set registeredFactoryNames() { return BEAN_FACTORIES.keySet(); } /** * Returns the default bean factory. * * @return the default bean factory */ public static BeanFactory getDefaultBeanFactory() { return defaultBeanFactory; } /** * Sets the default bean factory. This factory will be used for all create * operations, for which no special factory is provided in the bean * declaration. * * @param factory the default bean factory (must not be null) */ public static void setDefaultBeanFactory(BeanFactory factory) { if (factory == null) { throw new IllegalArgumentException( "Default bean factory must not be null!"); } defaultBeanFactory = factory; } /** * Initializes the passed in bean. This method will obtain all the bean's * properties that are defined in the passed in bean declaration. These * properties will be set on the bean. If necessary, further beans will be * created recursively. * * @param bean the bean to be initialized * @param data the bean declaration * @throws ConfigurationRuntimeException if a property cannot be set */ public static void initBean(Object bean, BeanDeclaration data) throws ConfigurationRuntimeException { initBeanProperties(bean, data); Map nestedBeans = data.getNestedBeanDeclarations(); if (nestedBeans != null) { if (bean instanceof Collection) { // This is safe because the collection stores the values of the // nested beans. @SuppressWarnings("unchecked") Collection coll = (Collection) bean; if (nestedBeans.size() == 1) { Map.Entry e = nestedBeans.entrySet().iterator().next(); String propName = e.getKey(); Class defaultClass = getDefaultClass(bean, propName); if (e.getValue() instanceof List) { // This is safe, provided that the bean declaration is implemented // correctly. @SuppressWarnings("unchecked") List decls = (List) e.getValue(); for (BeanDeclaration decl : decls) { coll.add(createBean(decl, defaultClass)); } } else { BeanDeclaration decl = (BeanDeclaration) e.getValue(); coll.add(createBean(decl, defaultClass)); } } } else { for (Map.Entry e : nestedBeans.entrySet()) { String propName = e.getKey(); Class defaultClass = getDefaultClass(bean, propName); Object prop = e.getValue(); if (prop instanceof Collection) { Collection beanCollection = createPropertyCollection(propName, defaultClass); for (Object elemDef : (Collection) prop) { beanCollection .add(createBean((BeanDeclaration) elemDef)); } initProperty(bean, propName, beanCollection); } else { initProperty(bean, propName, createBean( (BeanDeclaration) e.getValue(), defaultClass)); } } } } } /** * Initializes the beans properties. * * @param bean the bean to be initialized * @param data the bean declaration * @throws ConfigurationRuntimeException if a property cannot be set */ public static void initBeanProperties(Object bean, BeanDeclaration data) throws ConfigurationRuntimeException { Map properties = data.getBeanProperties(); if (properties != null) { for (Map.Entry e : properties.entrySet()) { String propName = e.getKey(); initProperty(bean, propName, e.getValue()); } } } /** * Return the Class of the property if it can be determined. * @param bean The bean containing the property. * @param propName The name of the property. * @return The class associated with the property or null. */ private static Class getDefaultClass(Object bean, String propName) { try { PropertyDescriptor desc = PropertyUtils.getPropertyDescriptor(bean, propName); if (desc == null) { return null; } return desc.getPropertyType(); } catch (Exception ex) { return null; } } /** * Sets a property on the given bean using Common Beanutils. * * @param bean the bean * @param propName the name of the property * @param value the property's value * @throws ConfigurationRuntimeException if the property is not writeable or * an error occurred */ private static void initProperty(Object bean, String propName, Object value) throws ConfigurationRuntimeException { if (!PropertyUtils.isWriteable(bean, propName)) { throw new ConfigurationRuntimeException("Property " + propName + " cannot be set on " + bean.getClass().getName()); } try { BeanUtils.setProperty(bean, propName, value); } catch (IllegalAccessException iaex) { throw new ConfigurationRuntimeException(iaex); } catch (InvocationTargetException itex) { throw new ConfigurationRuntimeException(itex); } } /** * Creates a concrete collection instance to populate a property of type * collection. This method tries to guess an appropriate collection type. * Mostly the type of the property will be one of the collection interfaces * rather than a concrete class; so we have to create a concrete equivalent. * * @param propName the name of the collection property * @param propertyClass the type of the property * @return the newly created collection */ private static Collection createPropertyCollection(String propName, Class propertyClass) { Collection beanCollection = null; if (List.class.isAssignableFrom(propertyClass)) { beanCollection = new ArrayList(); } else if (Set.class.isAssignableFrom(propertyClass)) { beanCollection = new TreeSet(); } else { throw new UnsupportedOperationException( "Unable to handle collection of type : " + propertyClass.getName() + " for property " + propName); } return beanCollection; } /** * Set a property on the bean only if the property exists * * @param bean the bean * @param propName the name of the property * @param value the property's value * @throws ConfigurationRuntimeException if the property is not writeable or * an error occurred */ public static void setProperty(Object bean, String propName, Object value) { if (PropertyUtils.isWriteable(bean, propName)) { initProperty(bean, propName, value); } } /** * The main method for creating and initializing beans from a configuration. * This method will return an initialized instance of the bean class * specified in the passed in bean declaration. If this declaration does not * contain the class of the bean, the passed in default class will be used. * From the bean declaration the factory to be used for creating the bean is * queried. The declaration may here return null, then a default * factory is used. This factory is then invoked to perform the create * operation. * * @param data the bean declaration * @param defaultClass the default class to use * @param param an additional parameter that will be passed to the bean * factory; some factories may support parameters and behave different * depending on the value passed in here * @return the new bean * @throws ConfigurationRuntimeException if an error occurs */ public static Object createBean(BeanDeclaration data, Class defaultClass, Object param) throws ConfigurationRuntimeException { if (data == null) { throw new IllegalArgumentException( "Bean declaration must not be null!"); } BeanFactory factory = fetchBeanFactory(data); try { return factory.createBean(fetchBeanClass(data, defaultClass, factory), data, param); } catch (Exception ex) { throw new ConfigurationRuntimeException(ex); } } /** * Returns a bean instance for the specified declaration. This method is a * short cut for {@code createBean(data, null, null);}. * * @param data the bean declaration * @param defaultClass the class to be used when in the declaration no class * is specified * @return the new bean * @throws ConfigurationRuntimeException if an error occurs */ public static Object createBean(BeanDeclaration data, Class defaultClass) throws ConfigurationRuntimeException { return createBean(data, defaultClass, null); } /** * Returns a bean instance for the specified declaration. This method is a * short cut for {@code createBean(data, null);}. * * @param data the bean declaration * @return the new bean * @throws ConfigurationRuntimeException if an error occurs */ public static Object createBean(BeanDeclaration data) throws ConfigurationRuntimeException { return createBean(data, null); } /** * Returns a {@code java.lang.Class} object for the specified name. * Because class loading can be tricky in some environments the code for * retrieving a class by its name was extracted into this helper method. So * if changes are necessary, they can be made at a single place. * * @param name the name of the class to be loaded * @param callingClass the calling class * @return the class object for the specified name * @throws ClassNotFoundException if the class cannot be loaded */ static Class loadClass(String name, Class callingClass) throws ClassNotFoundException { return ClassUtils.getClass(name); } /** * Determines the class of the bean to be created. If the bean declaration * contains a class name, this class is used. Otherwise it is checked * whether a default class is provided. If this is not the case, the * factory's default class is used. If this class is undefined, too, an * exception is thrown. * * @param data the bean declaration * @param defaultClass the default class * @param factory the bean factory to use * @return the class of the bean to be created * @throws ConfigurationRuntimeException if the class cannot be determined */ private static Class fetchBeanClass(BeanDeclaration data, Class defaultClass, BeanFactory factory) throws ConfigurationRuntimeException { String clsName = data.getBeanClassName(); if (clsName != null) { try { return loadClass(clsName, factory.getClass()); } catch (ClassNotFoundException cex) { throw new ConfigurationRuntimeException(cex); } } if (defaultClass != null) { return defaultClass; } Class clazz = factory.getDefaultBeanClass(); if (clazz == null) { throw new ConfigurationRuntimeException( "Bean class is not specified!"); } return clazz; } /** * Obtains the bean factory to use for creating the specified bean. This * method will check whether a factory is specified in the bean declaration. * If this is not the case, the default bean factory will be used. * * @param data the bean declaration * @return the bean factory to use * @throws ConfigurationRuntimeException if the factory cannot be determined */ private static BeanFactory fetchBeanFactory(BeanDeclaration data) throws ConfigurationRuntimeException { String factoryName = data.getBeanFactoryName(); if (factoryName != null) { BeanFactory factory = BEAN_FACTORIES.get(factoryName); if (factory == null) { throw new ConfigurationRuntimeException( "Unknown bean factory: " + factoryName); } else { return factory; } } else { return getDefaultBeanFactory(); } } }