blob: 02cc5f3c90c14780cf276af49d96283300a6cfcb [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2010, 2013 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
*******************************************************************************/
package org.eclipse.sisu.inject;
import java.lang.reflect.Method;
import java.util.Map;
import com.google.inject.Binding;
import com.google.inject.Injector;
import com.google.inject.Key;
import com.google.inject.TypeLiteral;
/**
* Publisher of {@link Binding}s from a single {@link Injector}; ranked according to a given {@link RankingFunction}.
*/
public final class InjectorPublisher
implements BindingPublisher
{
// ----------------------------------------------------------------------
// Static initialization
// ----------------------------------------------------------------------
static
{
Method getDeclaringSource;
try
{
// support future where binding.getSource() returns ElementSource and not the original declaring source
final Class<?> clazz = Binding.class.getClassLoader().loadClass( "com.google.inject.spi.ElementSource" );
getDeclaringSource = clazz.getMethod( "getDeclaringSource" );
}
catch ( final Exception e )
{
getDeclaringSource = null;
}
catch ( final LinkageError e )
{
getDeclaringSource = null;
}
GET_DECLARING_SOURCE = getDeclaringSource;
}
// ----------------------------------------------------------------------
// Constants
// ----------------------------------------------------------------------
private static final TypeLiteral<?> OBJECT_TYPE_LITERAL = TypeLiteral.get( Object.class );
private static final Method GET_DECLARING_SOURCE;
// ----------------------------------------------------------------------
// Implementation fields
// ----------------------------------------------------------------------
private final Injector injector;
private final RankingFunction function;
// ----------------------------------------------------------------------
// Constructors
// ----------------------------------------------------------------------
public InjectorPublisher( final Injector injector, final RankingFunction function )
{
this.injector = injector;
this.function = function;
}
public InjectorPublisher( final Injector injector )
{
this( injector, injector.getInstance( RankingFunction.class ) );
}
// ----------------------------------------------------------------------
// Public methods
// ----------------------------------------------------------------------
public Injector getInjector()
{
return injector;
}
public <T> void subscribe( final BindingSubscriber<T> subscriber )
{
final TypeLiteral<T> type = subscriber.type();
publishBindings( type, subscriber, null );
final Class<?> clazz = type.getRawType();
if ( clazz != type.getType() )
{
publishBindings( TypeLiteral.get( clazz ), subscriber, type );
}
if ( clazz != Object.class )
{
publishBindings( OBJECT_TYPE_LITERAL, subscriber, type );
}
}
public <T> void unsubscribe( final BindingSubscriber<T> subscriber )
{
final Map<Key<?>, ?> ourBindings = injector.getBindings();
for ( final Binding<T> binding : subscriber.bindings() )
{
if ( binding == ourBindings.get( binding.getKey() ) )
{
subscriber.remove( binding );
}
}
}
public int maxBindingRank()
{
return function.maxRank();
}
@Override
public int hashCode()
{
return injector.hashCode();
}
@Override
public boolean equals( final Object rhs )
{
if ( this == rhs )
{
return true;
}
if ( rhs instanceof InjectorPublisher )
{
return injector.equals( ( (InjectorPublisher) rhs ).injector );
}
return false;
}
@Override
public String toString()
{
return Logs.toString( injector );
}
// ----------------------------------------------------------------------
// Local methods
// ----------------------------------------------------------------------
static Object getDeclaringSource( final Binding<?> binding )
{
final Object source = binding.getSource();
if ( null != GET_DECLARING_SOURCE && GET_DECLARING_SOURCE.getDeclaringClass().isInstance( source ) )
{
try
{
return GET_DECLARING_SOURCE.invoke( source );
}
catch ( final Exception e )
{
// ignore
}
catch ( final LinkageError e )
{
// ignore
}
}
return source;
}
static boolean isVisible( final Binding<?> binding )
{
return false == getDeclaringSource( binding ) instanceof HiddenBinding;
}
// ----------------------------------------------------------------------
// Implementation methods
// ----------------------------------------------------------------------
private static boolean isAssignableFrom( final TypeLiteral<?> superType, final Binding<?> binding )
{
// don't match the exact implementation as it's already covered by an explicit binding
final Class<?> implementation = binding.acceptTargetVisitor( ImplementationVisitor.THIS );
if ( null != implementation && superType.getRawType() != implementation )
{
return TypeArguments.isAssignableFrom( superType, TypeLiteral.get( implementation ) );
}
return false;
}
@SuppressWarnings( { "rawtypes", "unchecked" } )
private void publishBindings( final TypeLiteral searchType, final BindingSubscriber subscriber,
final TypeLiteral superType )
{
for ( final Binding binding : injector.<Object> findBindingsByType( searchType ) )
{
if ( isVisible( binding ) && ( null == superType || isAssignableFrom( superType, binding ) ) )
{
subscriber.add( binding, function.rank( binding ) );
}
}
}
}