/*******************************************************************************
* Copyright (c) 2010-present Sonatype, Inc.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Stuart McCulloch (Sonatype, Inc.) - initial API and implementation
*
* Minimal facade required to be binary-compatible with legacy Plexus API
*******************************************************************************/
package org.codehaus.plexus.component.configurator.converters.composite;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.Map;
import java.util.Properties;
import java.util.TreeMap;
import org.codehaus.plexus.component.configurator.ComponentConfigurationException;
import org.codehaus.plexus.component.configurator.ConfigurationListener;
import org.codehaus.plexus.component.configurator.converters.AbstractConfigurationConverter;
import org.codehaus.plexus.component.configurator.converters.ConfigurationConverter;
import org.codehaus.plexus.component.configurator.converters.ParameterizedConfigurationConverter;
import org.codehaus.plexus.component.configurator.converters.lookup.ConverterLookup;
import org.codehaus.plexus.component.configurator.expression.ExpressionEvaluator;
import org.codehaus.plexus.configuration.PlexusConfiguration;
import org.eclipse.sisu.inject.Logs;
public class MapConverter
extends AbstractConfigurationConverter
implements ParameterizedConfigurationConverter
{
public boolean canConvert( final Class<?> type )
{
return Map.class.isAssignableFrom( type ) && !Properties.class.isAssignableFrom( type );
}
public Object fromConfiguration( final ConverterLookup lookup, final PlexusConfiguration configuration,
final Class<?> type, final Class<?> enclosingType, final ClassLoader loader,
final ExpressionEvaluator evaluator, final ConfigurationListener listener )
throws ComponentConfigurationException
{
return fromConfiguration( lookup, configuration, type, null, enclosingType, loader, evaluator, listener );
}
public Object fromConfiguration( final ConverterLookup lookup, final PlexusConfiguration configuration,
final Class<?> type, final Type[] typeArguments, final Class<?> enclosingType,
final ClassLoader loader, final ExpressionEvaluator evaluator,
final ConfigurationListener listener )
throws ComponentConfigurationException
{
final Object value = fromExpression( configuration, evaluator, type );
if ( null != value )
{
return value;
}
try
{
final Map<Object, Object> map = instantiateMap( configuration, type, loader );
final Class<?> elementType = findElementType( typeArguments );
if ( Object.class == elementType || String.class == elementType )
{
for ( int i = 0, size = configuration.getChildCount(); i < size; i++ )
{
final PlexusConfiguration element = configuration.getChild( i );
map.put( element.getName(), fromExpression( element, evaluator ) );
}
return map;
}
// handle maps with complex element types...
final ConfigurationConverter converter = lookup.lookupConverterForType( elementType );
for ( int i = 0, size = configuration.getChildCount(); i < size; i++ )
{
Object elementValue;
final PlexusConfiguration element = configuration.getChild( i );
try
{
elementValue = converter.fromConfiguration( lookup, element, elementType, enclosingType, //
loader, evaluator, listener );
}
// TEMP: remove when http://jira.codehaus.org/browse/MSHADE-168 is fixed
catch ( final ComponentConfigurationException e )
{
elementValue = fromExpression( element, evaluator );
Logs.warn( "Map in " + enclosingType + " declares value type as: {} but saw: {} at runtime",
elementType, null != elementValue ? elementValue.getClass() : null );
}
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
map.put( element.getName(), elementValue );
}
return map;
}
catch ( final ComponentConfigurationException e )
{
if ( null == e.getFailedConfiguration() )
{
e.setFailedConfiguration( configuration );
}
throw e;
}
}
@SuppressWarnings( "unchecked" )
private Map<Object, Object> instantiateMap( final PlexusConfiguration configuration, final Class<?> type,
final ClassLoader loader )
throws ComponentConfigurationException
{
final Class<?> implType = getClassForImplementationHint( type, configuration, loader );
if ( null == implType || Modifier.isAbstract( implType.getModifiers() ) )
{
return new TreeMap<Object, Object>();
}
final Object impl = instantiateObject( implType );
failIfNotTypeCompatible( impl, type, configuration );
return (Map<Object, Object>) impl;
}
private static Class<?> findElementType( final Type[] typeArguments )
{
if ( null != typeArguments && typeArguments.length > 1 && typeArguments[1] instanceof Class<?> )
{
return (Class<?>) typeArguments[1];
}
return Object.class;
}
}