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

org.apache.myfaces.trinidadinternal.convert.GenericConverterFactory Maven / Gradle / Ivy

There is a newer version: 2.2.1
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.myfaces.trinidadinternal.convert;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

import javax.faces.context.ExternalContext;
import javax.faces.context.FacesContext;

import org.apache.myfaces.trinidad.logging.TrinidadLogger;

/**
 * A factory of GenericConverters.
 * GenericConverters may be registered with this factory.
 * The factory supports converting between the types supported
 * by each individual converter. The factory also supports converting
 * between types supported by combining individual converters.
 * 
 */
public class GenericConverterFactory
{
  private GenericConverterFactory()
  {
    _cache = new ConcurrentHashMap(16);
    _converters = new ArrayList(3);
    _reverseDiscoveryConverters = new ArrayList(2);
    registerConverter(new SqlConverter());
    registerConverter(new BaseConverter());
    registerReverseDiscoveryConverter(new ReflectionConverter());
  }
  
  /**
   * Gets a converter that is capable of converting from the given
   * sourceType to the given targetType.
   * This method first searches to see if any of the registered converters
   * are capable of making this conversion alone. If one is found, it is
   * returned. Otherwise, this method will search to see if some combination
   * of registered converters can be used to perform this conversion.
   * If so, a composite converter will be returned.
   * 

* The returned converter (or lack thereof) is cached so that subsequent * requests for the same source and target types will be fast. * @return null if there is no such converter. */ public TypeConverter getConverter(Class sourceType, Class targetType) { Key key = new Key(sourceType, targetType); // check the cache first: Object cached = _cache.get(key); if (cached != null) { return (cached == _NULL) ? null : (TypeConverter) cached; } // we are going to start searching to see if some chain of converters // can be used to perform this conversion. // initial node in chain: Node start = new Node(null, null, sourceType); LinkedList sourcesToBeSearched = new LinkedList(); sourcesToBeSearched.add(start); // cache to store all the classes we've tested already. This is to // avoid our chains from looping indefinitely: Set> cache = new HashSet>(16); // Try to find a converter chain without "reverse discovery" type converetrs. // Our particular implementation of the "reverse discovery" type converter uses Java Reflection, // so it is better to use a chain of normal generic converters TypeConverter converter = _findConverter(sourcesToBeSearched, targetType, cache, false); // If we failed, try to use "reverse discovery" converters if (converter == null && _reverseDiscoveryConverters.size() > 0) { // We will use a "reverse discovery" type converter only if target-to-source conversion // is available without the use of "reverse discovery" type converters. This is done to ensure // we do not find a converter dynamically for the target type when the source type is // not convertable from the target type (converter's availability has to be 'reflective'). // Regular generic converters are pre-defined, so they normally do not have this problem. TypeConverter reverseConv = null; reverseConv = _cache.get(new Key(targetType, sourceType)); if (reverseConv == null) { cache.clear(); sourcesToBeSearched.add( new Node(null, null, targetType)); reverseConv = _findConverter(sourcesToBeSearched, sourceType, cache, false); } if (reverseConv != null) { cache.clear(); sourcesToBeSearched.clear(); sourcesToBeSearched.add(start); converter = _findConverter(sourcesToBeSearched, targetType, cache, true); } } if (converter == null) { // cache the fact that no such converter exists: _cache.put(key, _NULL); } else { _cache.put(key, converter); } return converter; } /** * tries to find a converter, or create a chain of converters that can * convert from one of the given sourceTypes to the specified targetType. * @param sourcesToBeSearched each element is a Node. Each Node is a pairing of * sourceType and the chain of converters needed to produce this sourceType. * @param targetType the type that is needed * @param cache used to record which classes we've searched already. * @param useRevserseDiscovery true if "reverse discovery" converters should be used, false otherwise * @return null if no converter was found. */ private TypeConverter _findConverter( LinkedList sourcesToBeSearched, Class targetType, Set> cache, boolean useRevserseDiscovery) { while(!sourcesToBeSearched.isEmpty()) { Node source = sourcesToBeSearched.removeFirst(); TypeConverter match = null; // loop through all the converters and see what types they can turn // the current sourceType into // (the current sourceType is source.targetType): for(int j=0,jsz=_converters.size(); j sourcesToBeSearched, Node currentSource, GenericConverter currentConverter, Class searchType, Set> cache) { Class sourceType = currentSource.targetType; List> targetTypes = currentConverter.getTargetTypes(sourceType); for(int i=0,sz=targetTypes.size(); i targetType = targetTypes.get(i); // check to see if we've seen this targetType before: if (cache.add(targetType)) { // check to see if the targetType is a subclass of the searchType: if (searchType.isAssignableFrom(targetType)) return true; // create a new node in the chain by adding this targetType and converter Node newSource = new Node(currentSource, currentConverter, targetType); // add the new node so that we can continue searching by seeing if // we can convert the targetType into the searchType using some other // converter: sourcesToBeSearched.add(newSource); } } return false; } /** * Finds a suitable converter by searching source types supported for a * given target type * @param sourceType - source type * @param targetType - target type * @return a suitable TypeConverter is found, null otherwise */ private TypeConverter _searchSourceTypes(Class sourceType, Class targetType) { for (ReverseDiscoveryGenericConverter conv:_reverseDiscoveryConverters) { List> sourceTypes = conv.getSourceTypes(targetType); for (Class type: sourceTypes) { if (type.isAssignableFrom(sourceType)) return conv; } } return null; } /** * Registers a converter. Registering a new converter causes the internal * cache to be cleared. */ public void registerConverter(GenericConverter converter) { _converters.add(converter); _cache.clear(); } /** * Registers a "reverse discovery" converter. Registering a new converter causes the internal * cache to be cleared. */ public void registerReverseDiscoveryConverter(ReverseDiscoveryGenericConverter converter) { _reverseDiscoveryConverters.add(converter); _cache.clear(); } /** * converts the given source instance into an object of the targetType. * @param source the object to convert * @param targetType the required type. * @return null, if the source is null. */ public Object convert(Object source, Class targetType) { if (source == null) return null; if (targetType.isAssignableFrom(source.getClass())) return source; TypeConverter converter = getConverter(source.getClass(), targetType); if (converter != null) { try { return converter.convert(source, targetType); } catch(Exception exe) { // Rethrow any resulting exception wrapped in a TypeConversion exception throw new TypeConversionException(source, targetType, exe); } } throw new TypeConversionException(source, targetType); } /** * Checks to see if it is possible to convert the given instance * into the specified targetType * @return true if conversion is possible. */ public boolean isConvertible(Object source, Class targetType) { if (source == null) return false; // bug 4589048 if (targetType.isAssignableFrom(source.getClass())) return true; TypeConverter converter = getConverter(source.getClass(), targetType); return converter != null; } private final Map _cache; private final List _converters; private final List _reverseDiscoveryConverters; // 2006-08-02: -= Simon Lessard =- // Using a GenericConverter null value instead // of Node.class to be typesafe private static final TypeConverter _NULL = new TypeConverter() { public Object convert(Object source, Class targetType) { return null; } }; private static final class Node { public Node(Node previous, TypeConverter converter, Class targetType) { this.previous = previous; this.converter = converter; this.targetType = targetType; } public Object convert(Object source) { if (previous != null) { source = previous.convert(source); source = converter.convert(source, targetType); } return source; } public final Node previous; public final TypeConverter converter; public final Class targetType; } private static final class Key { public Key(Class source, Class target) { assert !source.equals(target); _source = source; _target = target; _hc = source.hashCode() + target.hashCode(); } @Override public int hashCode() { return _hc; } @Override public boolean equals(Object other) { if (this == other) return true; if (other instanceof Key) { Key that = (Key) other; return (_source.equals(that._source) && _target.equals(that._target)); } return false; } private final int _hc; private final Class _source; private final Class _target; } private static final class CompositeConverter implements TypeConverter { public CompositeConverter(Node source, TypeConverter conv, Class targetType) { assert source != null; _chain = new Node(source, conv, targetType) ; } public Object convert(Object source, Class targetType) { if (targetType.isAssignableFrom(_chain.targetType)) { return _chain.convert(source); } else throw new IllegalArgumentException(_LOG.getMessage( "CANNOT_CONVERT", new Object[]{source, targetType.getName()})); } private final Node _chain; } public static GenericConverterFactory getCurrentInstance(ExternalContext extContext) { // TODO: once getCurrentInstance() taking ExternalContext is in ADE, // we need to modify FacesDatabindingConfigurator to pass in the context. // Then we can switch to storing factory instance on the application map /*if (extContext == null) extContext = FacesContext.getCurrentInstance().getExternalContext(); Map appMap = extContext.getApplicationMap(); GenericConverterFactory factory = (GenericConverterFactory)appMap.get(_INSTANCE_KEY); if (factory == null) { factory = new GenericConverterFactory(); appMap.put(_INSTANCE_KEY, factory); } return factory;*/ return _INSTANCE; } public static GenericConverterFactory getCurrentInstance() { return getCurrentInstance(null); } private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger( GenericConverterFactory.class); //private static final String _INSTANCE_KEY = "org.apache.myfaces.trinidadinternal.convert.GenericConverterFactory"; private static final GenericConverterFactory _INSTANCE = new GenericConverterFactory(); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy