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

org.apache.xbean.propertyeditor.PropertyEditors Maven / Gradle / Ivy

/**
 * 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.xbean.propertyeditor;

import static org.apache.xbean.recipe.RecipeHelper.getTypeParameters;
import static org.apache.xbean.recipe.RecipeHelper.*;
import org.apache.xbean.recipe.RecipeHelper;

import java.beans.PropertyEditor;
import java.beans.PropertyEditorManager;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.Collection;
import java.util.SortedSet;
import java.util.Set;
import java.util.TreeSet;
import java.util.LinkedHashSet;
import java.util.ArrayList;
import java.util.SortedMap;
import java.util.TreeMap;
import java.util.LinkedHashMap;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ConcurrentHashMap;
import java.lang.reflect.Type;

/**
 * The property editor manager.  This orchestrates Geronimo usage of
 * property editors, allowing additional search paths to be added and
 * specific editors to be registered.
 *
 * @version $Rev: 6687 $
 */
public class PropertyEditors {
    private static final Map registry = Collections.synchronizedMap(new ReferenceIdentityMap());
    private static final Map PRIMITIVE_TO_WRAPPER;
    private static final Map WRAPPER_TO_PRIMITIVE;
    private static boolean registerWithVM;

    /**
     * Register all of the built in converters
     */
    static {
        Map map = new HashMap();
        map.put(boolean.class, Boolean.class);
        map.put(char.class, Character.class);
        map.put(byte.class, Byte.class);
        map.put(short.class, Short.class);
        map.put(int.class, Integer.class);
        map.put(long.class, Long.class);
        map.put(float.class, Float.class);
        map.put(double.class, Double.class);
        PRIMITIVE_TO_WRAPPER = Collections.unmodifiableMap(map);


        map = new HashMap();
        map.put(Boolean.class, boolean.class);
        map.put(Character.class, char.class);
        map.put(Byte.class, byte.class);
        map.put(Short.class, short.class);
        map.put(Integer.class, int.class);
        map.put(Long.class, long.class);
        map.put(Float.class, float.class);
        map.put(Double.class, double.class);
        WRAPPER_TO_PRIMITIVE = Collections.unmodifiableMap(map);

        // Explicitly register the types
        registerConverter(new ArrayListEditor());
        registerConverter(new BigDecimalEditor());
        registerConverter(new BigIntegerEditor());
        registerConverter(new BooleanEditor());
        registerConverter(new ByteEditor());
        registerConverter(new CharacterEditor());
        registerConverter(new ClassEditor());
        registerConverter(new DateEditor());
        registerConverter(new DoubleEditor());
        registerConverter(new FileEditor());
        registerConverter(new FloatEditor());
        registerConverter(new HashMapEditor());
        registerConverter(new HashtableEditor());
        registerConverter(new IdentityHashMapEditor());
        registerConverter(new Inet4AddressEditor());
        registerConverter(new Inet6AddressEditor());
        registerConverter(new InetAddressEditor());
        registerConverter(new IntegerEditor());
        registerConverter(new LinkedHashMapEditor());
        registerConverter(new LinkedHashSetEditor());
        registerConverter(new LinkedListEditor());
        registerConverter(new ListEditor());
        registerConverter(new LongEditor());
        registerConverter(new MapEditor());
        registerConverter(new ObjectNameEditor());
        registerConverter(new PropertiesEditor());
        registerConverter(new SetEditor());
        registerConverter(new ShortEditor());
        registerConverter(new SortedMapEditor());
        registerConverter(new SortedSetEditor());
        registerConverter(new StringEditor());
        registerConverter(new TreeMapEditor());
        registerConverter(new TreeSetEditor());
        registerConverter(new URIEditor());
        registerConverter(new URLEditor());
        registerConverter(new LoggerConverter());
        registerConverter(new PatternConverter());
        registerConverter(new JndiConverter());
        registerConverter(new VectorEditor());
        registerConverter(new WeakHashMapEditor());

        try {
            registerConverter(new Log4jConverter());
        } catch (Throwable e) {
        }

        try {
            registerConverter(new CommonsLoggingConverter());
        } catch (Throwable e) {
        }
    }

    /**
     * Are converters registered with the VM PropertyEditorManager.  By default
     * converters are not registered with the VM as this creates problems for
     * IDE and Spring because they rely in their specific converters being
     * registered to function properly. 
     */
    public static boolean isRegisterWithVM() {
        return registerWithVM;
    }

    /**
     * Sets if converters registered with the VM PropertyEditorManager.
     * If the new value is true, all currently registered converters are
     * immediately registered with the VM.
     */
    public static void setRegisterWithVM(boolean registerWithVM) {
        if (PropertyEditors.registerWithVM != registerWithVM) {
            PropertyEditors.registerWithVM = registerWithVM;

            // register all converters with the VM
            if (registerWithVM) {
                for (Entry entry : registry.entrySet()) {
                    Class type = entry.getKey();
                    Converter converter = entry.getValue();
                    PropertyEditorManager.registerEditor(type, converter.getClass());
                }
            }
        }
    }

    public static void registerConverter(Converter converter) {
        if (converter == null) throw new NullPointerException("editor is null");
        Class type = converter.getType();
        registry.put(type, converter);
        if (registerWithVM) {
            PropertyEditorManager.registerEditor(type, converter.getClass());
        }

        if (PRIMITIVE_TO_WRAPPER.containsKey(type)) {
            Class wrapperType = PRIMITIVE_TO_WRAPPER.get(type);
            registry.put(wrapperType, converter);
            if (registerWithVM) {
                PropertyEditorManager.registerEditor(wrapperType, converter.getClass());
            }
        } else if (WRAPPER_TO_PRIMITIVE.containsKey(type)) {
            Class primitiveType = WRAPPER_TO_PRIMITIVE.get(type);
            registry.put(primitiveType, converter);
            if (registerWithVM) {
                PropertyEditorManager.registerEditor(primitiveType, converter.getClass());
            }
        }
    }

    public static boolean canConvert(String type, ClassLoader classLoader) {
        if (type == null) throw new NullPointerException("type is null");
        if (classLoader == null) throw new NullPointerException("classLoader is null");

        // load using the ClassLoading utility, which also manages arrays and primitive classes.
        Class typeClass;
        try {
            typeClass = Class.forName(type, true, classLoader);
        } catch (ClassNotFoundException e) {
            throw new PropertyEditorException("Type class could not be found: " + type);
        }

        return canConvert(typeClass);

    }

    public static boolean canConvert(Class type) {
        PropertyEditor editor = findConverterOrEditor(type);

        return editor != null;
    }

    private static PropertyEditor findConverterOrEditor(Type type){
        Converter converter = findConverter(type);
        if (converter != null) {
            return converter;
        }

        // fall back to a property editor
        PropertyEditor editor = findEditor(type);
        if (editor != null) {
            return editor;
        }

        converter = findBuiltinConverter(type);
        if (converter != null) {
            return converter;
        }

        return null;
    }

    public static String toString(Object value) throws PropertyEditorException {
        if (value == null) throw new NullPointerException("value is null");

        // get an editor for this type
        Class type = value.getClass();

        PropertyEditor editor = findConverterOrEditor(type);

        if (editor instanceof Converter) {
            Converter converter = (Converter) editor;
            return converter.toString(value);
        }

        if (editor == null) {
            throw new PropertyEditorException("Unable to find PropertyEditor for " + type.getSimpleName());
        }

        // create the string value
        editor.setValue(value);
        String textValue;
        try {
            textValue = editor.getAsText();
        } catch (Exception e) {
            throw new PropertyEditorException("Error while converting a \"" + type.getSimpleName() + "\" to text " +
                    " using the property editor " + editor.getClass().getSimpleName(), e);
        }
        return textValue;
    }

    public static Object getValue(String type, String value, ClassLoader classLoader) throws PropertyEditorException {
        if (type == null) throw new NullPointerException("type is null");
        if (value == null) throw new NullPointerException("value is null");
        if (classLoader == null) throw new NullPointerException("classLoader is null");

        // load using the ClassLoading utility, which also manages arrays and primitive classes.
        Class typeClass;
        try {
            typeClass = Class.forName(type, true, classLoader);
        } catch (ClassNotFoundException e) {
            throw new PropertyEditorException("Type class could not be found: " + type);
        }

        return getValue(typeClass, value);

    }

    public static Object getValue(Type type, String value) throws PropertyEditorException {
        if (type == null) throw new NullPointerException("type is null");
        if (value == null) throw new NullPointerException("value is null");

        PropertyEditor editor = findConverterOrEditor(type);

        if (editor instanceof Converter) {
            Converter converter = (Converter) editor;
            return converter.toObject(value);
        }

        Class clazz = toClass(type);

        if (editor == null) {
            throw new PropertyEditorException("Unable to find PropertyEditor for " + clazz.getSimpleName());
        }

        editor.setAsText(value);
        Object objectValue;
        try {
            objectValue = editor.getValue();
        } catch (Exception e) {
            throw new PropertyEditorException("Error while converting \"" + value + "\" to a " + clazz.getSimpleName() +
                    " using the property editor " + editor.getClass().getSimpleName(), e);
        }
        return objectValue;
    }

    private static Converter findBuiltinConverter(Type type) {
        if (type == null) throw new NullPointerException("type is null");

        Class clazz = toClass(type);

        if (Enum.class.isAssignableFrom(clazz)){
            return new EnumConverter(clazz);
        }

        return null;       
    }

    private static Converter findConverter(Type type) {
        if (type == null) throw new NullPointerException("type is null");

        Class clazz = toClass(type);



        // it's possible this was a request for an array class.  We might not
        // recognize the array type directly, but the component type might be
        // resolvable
        if (clazz.isArray() && !clazz.getComponentType().isArray()) {
            // do a recursive lookup on the base type
            PropertyEditor editor = findConverterOrEditor(clazz.getComponentType());
            // if we found a suitable editor for the base component type,
            // wrapper this in an array adaptor for real use
            if (editor != null) {
                return new ArrayConverter(clazz, editor);
            } else {
                return null;
            }
        }

        if (Collection.class.isAssignableFrom(clazz)){
            Type[] types = getTypeParameters(Collection.class, type);

            Type componentType = String.class;
            if (types != null && types.length == 1 && types[0] instanceof Class) {
                componentType = types[0];
            }

            PropertyEditor editor = findConverterOrEditor(componentType);

            if (editor != null){
                if (RecipeHelper.hasDefaultConstructor(clazz)) {
                    return new GenericCollectionConverter(clazz, editor);
                } else if (SortedSet.class.isAssignableFrom(clazz)) {
                    return new GenericCollectionConverter(TreeSet.class, editor);
                } else if (Set.class.isAssignableFrom(clazz)) {
                    return new GenericCollectionConverter(LinkedHashSet.class, editor);
                } else {
                    return new GenericCollectionConverter(ArrayList.class, editor);
                }
            }

            return null;
        }

        if (Map.class.isAssignableFrom(clazz)){
            Type[] types = getTypeParameters(Map.class, type);

            Type keyType = String.class;
            Type valueType = String.class;
            if (types != null && types.length == 2 && types[0] instanceof Class && types[1] instanceof Class) {
                keyType = types[0];
                valueType = types[1];
            }

            PropertyEditor keyConverter = findConverterOrEditor(keyType);
            PropertyEditor valueConverter = findConverterOrEditor(valueType);

            if (keyConverter != null && valueConverter != null){
                if (RecipeHelper.hasDefaultConstructor(clazz)) {
                    return new GenericMapConverter(clazz, keyConverter, valueConverter);
                } else if (SortedMap.class.isAssignableFrom(clazz)) {
                    return new GenericMapConverter(TreeMap.class, keyConverter, valueConverter);
                } else if (ConcurrentMap.class.isAssignableFrom(clazz)) {
                    return new GenericMapConverter(ConcurrentHashMap.class, keyConverter, valueConverter);
                } else {
                    return new GenericMapConverter(LinkedHashMap.class, keyConverter, valueConverter);
                }
            }

            return null;
        }

        Converter converter = registry.get(clazz);

        // we're outta here if we got one.
        if (converter != null) {
            return converter;
        }

        Class[] declaredClasses = clazz.getDeclaredClasses();
        for (Class declaredClass : declaredClasses) {
            if (Converter.class.isAssignableFrom(declaredClass)) {
                try {
                    converter = (Converter) declaredClass.newInstance();
                    registerConverter(converter);

                    // try to get the converter from the registry... the converter
                    // created above may have been for another class
                    converter = registry.get(clazz);
                    if (converter != null) {
                        return converter;
                    }
                } catch (Exception e) {
                }

            }
        }

        // nothing found
        return null;
    }

    /**
     * Locate a property editor for qiven class of object.
     *
     * @param type The target object class of the property.
     * @return The resolved editor, if any.  Returns null if a suitable editor
     *         could not be located.
     */
    private static PropertyEditor findEditor(Type type) {
        if (type == null) throw new NullPointerException("type is null");

        Class clazz = toClass(type);

        // try to locate this directly from the editor manager first.
        PropertyEditor editor = PropertyEditorManager.findEditor(clazz);

        // we're outta here if we got one.
        if (editor != null) {
            return editor;
        }


        // it's possible this was a request for an array class.  We might not
        // recognize the array type directly, but the component type might be
        // resolvable
        if (clazz.isArray() && !clazz.getComponentType().isArray()) {
            // do a recursive lookup on the base type
            editor = findEditor(clazz.getComponentType());
            // if we found a suitable editor for the base component type,
            // wrapper this in an array adaptor for real use
            if (editor != null) {
                return new ArrayConverter(clazz, editor);
            }
        }

        // nothing found
        return null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy