/******************************************************************************* * 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 *******************************************************************************/ package org.eclipse.sisu.inject; import org.eclipse.sisu.Hidden; import com.google.inject.Binding; import com.google.inject.ImplementedBy; import com.google.inject.Injector; import com.google.inject.Key; import com.google.inject.ProvidedBy; import com.google.inject.TypeLiteral; /** * Source of "implicit" bindings; includes @{@link ImplementedBy}, @{@link ProvidedBy}, and concrete types. */ final class ImplicitBindings { // ---------------------------------------------------------------------- // Implementation fields // ---------------------------------------------------------------------- private final Iterable<BindingPublisher> publishers; // ---------------------------------------------------------------------- // Constructors // ---------------------------------------------------------------------- ImplicitBindings( final Iterable<BindingPublisher> publishers ) { this.publishers = publishers; } // ---------------------------------------------------------------------- // Public methods // ---------------------------------------------------------------------- @SuppressWarnings( { "unchecked", "rawtypes" } ) public <T> Binding<T> get( final TypeLiteral<T> type ) { // first round: check for any re-written implicit bindings final Key implicitKey = TypeArguments.implicitKey( type.getRawType() ); for ( final BindingPublisher p : publishers ) { final Injector injector = p.adapt( Injector.class ); if ( null != injector ) { final Binding binding = injector.getBindings().get( implicitKey ); if ( null != binding ) { Logs.trace( "Using implicit binding: {} from: <>", binding, injector ); return binding; } } } // second round: fall back to just-in-time binding lookup final Key justInTimeKey = Key.get( type ); for ( final BindingPublisher p : publishers ) { final Injector injector = p.adapt( Injector.class ); if ( null != injector ) { try { final Binding binding = injector.getBinding( justInTimeKey ); if ( null == Sources.getAnnotation( binding, Hidden.class ) ) { Logs.trace( "Using just-in-time binding: {} from: <>", binding, injector ); return binding; } } catch ( final RuntimeException e ) { Logs.trace( "Problem with just-in-time binding: {}", justInTimeKey, e ); } catch ( final LinkageError e ) { Logs.trace( "Problem with just-in-time binding: {}", justInTimeKey, e ); } } } return null; } }