blob: 8d87bb99d63c76e296355dd073c53e0614049239 [file] [log] [blame]
/*
* Copyright (c) 2007, Rickard Öberg. All Rights Reserved.
*
* Licensed 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.zest.runtime.injection.provider;
import java.lang.annotation.Annotation;
import java.util.HashMap;
import java.util.Map;
import org.apache.zest.api.common.MetaInfo;
import org.apache.zest.api.composite.InvalidValueCompositeException;
import org.apache.zest.api.composite.ModelDescriptor;
import org.apache.zest.api.concern.internal.ConcernFor;
import org.apache.zest.api.injection.scope.Invocation;
import org.apache.zest.api.injection.scope.Service;
import org.apache.zest.api.injection.scope.State;
import org.apache.zest.api.injection.scope.Structure;
import org.apache.zest.api.injection.scope.This;
import org.apache.zest.api.injection.scope.Uses;
import org.apache.zest.api.sideeffect.internal.SideEffectFor;
import org.apache.zest.api.value.ValueComposite;
import org.apache.zest.bootstrap.InvalidInjectionException;
import org.apache.zest.runtime.injection.DependencyModel;
import org.apache.zest.runtime.injection.InjectionProvider;
import org.apache.zest.runtime.injection.InjectionProviderFactory;
import org.apache.zest.runtime.model.Resolution;
import static org.apache.zest.functional.Iterables.first;
/**
* JAVADOC
*/
public final class InjectionProviderFactoryStrategy
implements InjectionProviderFactory
{
private final Map<Class<? extends Annotation>, InjectionProviderFactory> generalProviderFactories = new HashMap<Class<? extends Annotation>, InjectionProviderFactory>();
private final Map<Class<? extends Annotation>, InjectionProviderFactory> valuesProviderFactories = new HashMap<Class<? extends Annotation>, InjectionProviderFactory>();
private MetaInfo metaInfo;
public InjectionProviderFactoryStrategy( MetaInfo metaInfo )
{
this.metaInfo = metaInfo;
valuesProviderFactories.put( This.class, new ThisInjectionProviderFactory() );
ModifiesInjectionProviderFactory modifiesInjectionProviderFactory = new ModifiesInjectionProviderFactory();
valuesProviderFactories.put( ConcernFor.class, modifiesInjectionProviderFactory );
valuesProviderFactories.put( SideEffectFor.class, modifiesInjectionProviderFactory );
valuesProviderFactories.put( State.class, new StateInjectionProviderFactory() );
valuesProviderFactories.put( Structure.class, new CachingInjectionProviderFactoryDecorator( new StructureInjectionProviderFactory() ) );
valuesProviderFactories.put( Service.class, new CachingInjectionProviderFactoryDecorator( new ServiceInjectionProviderFactory() ) );
generalProviderFactories.put( Invocation.class, new InvocationInjectionProviderFactory() );
generalProviderFactories.put( Uses.class, new UsesInjectionProviderFactory() );
}
@Override
public InjectionProvider newInjectionProvider( Resolution resolution, DependencyModel dependencyModel )
throws InvalidInjectionException
{
Class<? extends Annotation> injectionAnnotationType = dependencyModel.injectionAnnotation().annotationType();
InjectionProviderFactory factory1 = generalProviderFactories.get( injectionAnnotationType );
InjectionProviderFactory factory2 = valuesProviderFactories.get( injectionAnnotationType );
if( factory1 == null && factory2 == null )
{
InjectionProviderFactory factory = metaInfo.get( InjectionProviderFactory.class );
if( factory != null )
{
return factory.newInjectionProvider( resolution, dependencyModel );
}
else
{
throw new InvalidInjectionException( "Unknown injection annotation @" + injectionAnnotationType.getSimpleName() );
}
}
ModelDescriptor composite = resolution.model();
Class<?> compositeType = first( composite.types() );
if( factory1 != null && ValueComposite.class.isAssignableFrom( compositeType ) )
{
throw new InvalidValueCompositeException( "@" + injectionAnnotationType.getSimpleName() + " is not allowed in ValueComposites: " + compositeType );
}
InjectionProviderFactory factory;
if( factory1 == null )
{
factory = factory2;
}
else
{
factory = factory1;
}
return factory.newInjectionProvider( resolution, dependencyModel );
}
}