blob: fa720377b08b6be0837f7c1c92d26e2d1a2b8c1a [file] [log] [blame]
// Copyright 2006, 2007, 2008, 2009, 2010 The Apache Software Foundation
//
// 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.tapestry5.ioc.internal;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Map;
import java.util.Set;
import org.apache.tapestry5.ioc.Configuration;
import org.apache.tapestry5.ioc.MappedConfiguration;
import org.apache.tapestry5.ioc.ModuleBuilderSource;
import org.apache.tapestry5.ioc.ObjectLocator;
import org.apache.tapestry5.ioc.OrderedConfiguration;
import org.apache.tapestry5.ioc.ServiceResources;
import org.apache.tapestry5.ioc.def.ContributionDef2;
import org.apache.tapestry5.ioc.internal.util.CollectionFactory;
import org.apache.tapestry5.ioc.internal.util.DelegatingInjectionResources;
import org.apache.tapestry5.ioc.internal.util.InjectionResources;
import org.apache.tapestry5.ioc.internal.util.InternalUtils;
import org.apache.tapestry5.ioc.internal.util.MapInjectionResources;
import org.apache.tapestry5.ioc.internal.util.WrongConfigurationTypeGuard;
import org.apache.tapestry5.ioc.services.ClassFactory;
import org.slf4j.Logger;
public class ContributionDefImpl implements ContributionDef2
{
private final String serviceId;
private final Method contributorMethod;
private final ClassFactory classFactory;
private final Set<Class> markers;
private final Class serviceInterface;
private static final Class[] CONFIGURATION_TYPES = new Class[] {Configuration.class, MappedConfiguration.class,
OrderedConfiguration.class};
public ContributionDefImpl(String serviceId, Method contributorMethod, ClassFactory classFactory, Class serviceInterface, Set<Class> markers)
{
this.serviceId = serviceId;
this.contributorMethod = contributorMethod;
this.classFactory = classFactory;
this.serviceInterface = serviceInterface;
this.markers = markers;
}
@Override
public String toString()
{
return InternalUtils.asString(contributorMethod, classFactory);
}
public String getServiceId()
{
return serviceId;
}
public void contribute(ModuleBuilderSource moduleSource, ServiceResources resources,
Configuration configuration)
{
invokeMethod(moduleSource, resources, Configuration.class, configuration);
}
public void contribute(ModuleBuilderSource moduleSource, ServiceResources resources,
OrderedConfiguration configuration)
{
invokeMethod(moduleSource, resources, OrderedConfiguration.class, configuration);
}
public void contribute(ModuleBuilderSource moduleSource, ServiceResources resources,
MappedConfiguration configuration)
{
invokeMethod(moduleSource, resources, MappedConfiguration.class, configuration);
}
private <T> void invokeMethod(ModuleBuilderSource source, ServiceResources resources,
Class<T> parameterType, T parameterValue)
{
Map<Class, Object> resourceMap = CollectionFactory.newMap();
resourceMap.put(parameterType, parameterValue);
resourceMap.put(ObjectLocator.class, resources);
resourceMap.put(Logger.class, resources.getLogger());
InjectionResources injectionResources = new MapInjectionResources(resourceMap);
// For each of the other configuration types that is not expected, add a guard.
for (Class t : CONFIGURATION_TYPES)
{
if (parameterType != t)
{
injectionResources = new DelegatingInjectionResources(
new WrongConfigurationTypeGuard(resources.getServiceId(), t, parameterType),
injectionResources);
}
}
Throwable fail = null;
Object moduleInstance = InternalUtils.isStatic(contributorMethod) ? null : source
.getModuleBuilder();
try
{
Object[] parameters = InternalUtils.calculateParametersForMethod(
contributorMethod,
resources,
injectionResources, resources.getTracker());
contributorMethod.invoke(moduleInstance, parameters);
}
catch (InvocationTargetException ex)
{
fail = ex.getTargetException();
}
catch (Exception ex)
{
fail = ex;
}
if (fail != null)
throw new RuntimeException(IOCMessages
.contributionMethodError(contributorMethod, fail), fail);
}
public Set<Class> getMarkers()
{
return markers;
}
public Class getServiceInterface()
{
return serviceInterface;
}
}