org.apache.webbeans.component.InterceptorBean Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.webbeans.component;
import javax.enterprise.context.Dependent;
import javax.enterprise.inject.spi.AnnotatedType;
import javax.enterprise.inject.spi.InterceptionType;
import javax.enterprise.inject.spi.Interceptor;
import javax.interceptor.InvocationContext;
import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Collections;
import java.util.Map;
import javax.enterprise.inject.spi.BeanAttributes;
import org.apache.webbeans.config.WebBeansContext;
import org.apache.webbeans.container.InjectionTargetFactoryImpl;
import org.apache.webbeans.container.InterceptorInjectionTargetFactory;
import org.apache.webbeans.util.ExceptionUtil;
/**
* Abstract {@link javax.enterprise.inject.spi.Interceptor} Bean implementation.
*
*
Any Interceptor is also an InjectionTarget as they can contain
* @Inject InjectionPoints.
*/
public abstract class InterceptorBean extends InjectionTargetBean implements Interceptor
{
/**
* The Methods to be called per interception type.
* The method signature must be
* Object (InvocationContext) throws Exception
*/
private Map interceptionMethods;
/**
* This is for performance reasons
*/
protected Method aroundInvokeMethod = null;
protected InterceptorBean(WebBeansContext webBeansContext,
AnnotatedType annotatedType,
BeanAttributes beanAttributes,
Class beanClass,
Map interceptionMethods,
InjectionTargetFactoryImpl factory)
{
super(webBeansContext,
WebBeansType.INTERCEPTOR,
annotatedType,
beanAttributes,
beanClass,
factory);
this.interceptionMethods = Collections.unmodifiableMap(interceptionMethods);
for (Method[] methods: interceptionMethods.values())
{
for (Method method: methods)
{
if (!method.isAccessible())
{
method.setAccessible(true);
}
}
}
Method[] aroundInvokeMethods = interceptionMethods.get(InterceptionType.AROUND_INVOKE);
if (aroundInvokeMethods != null && aroundInvokeMethods.length == 1)
{
aroundInvokeMethod = aroundInvokeMethods[0];
}
}
public InterceptorBean(WebBeansContext webBeansContext,
AnnotatedType annotatedType,
BeanAttributes beanAttributes,
Class beanClass,
Map interceptionMethods)
{
this(webBeansContext, annotatedType, beanAttributes, beanClass, interceptionMethods,
new InterceptorInjectionTargetFactory(annotatedType, webBeansContext));
}
/**
* Interceptors are by default @Dependent scoped.
*/
@Override
public Class extends Annotation> getScope()
{
return Dependent.class;
}
/**
* @param interceptionType
* @return the underlying interceptor method for the given InterceptionType or null
*/
public Method[] getInterceptorMethods(InterceptionType interceptionType)
{
return interceptionMethods.get(interceptionType);
}
@Override
public boolean intercepts(InterceptionType interceptionType)
{
return interceptionMethods.containsKey(interceptionType);
}
@Override
public Object intercept(InterceptionType interceptionType, T instance, InvocationContext invocationContext)
{
try
{
if (InterceptionType.AROUND_INVOKE == interceptionType && aroundInvokeMethod != null)
{
return aroundInvokeMethod.invoke(instance, invocationContext);
}
Method[] interceptorMethods = getInterceptorMethods(interceptionType);
if (interceptorMethods == null || interceptorMethods.length == 0)
{
// this very interceptor doesn't support this interception type.
// this might happen for lifecycle callback methods
// let's continue with the next interceptor
return invocationContext.proceed();
}
else if (interceptorMethods.length == 1)
{
// directly invoke the interceptor method with the given InvocationContext
if (interceptorMethods[0].getParameterTypes().length == 1)
{
return interceptorMethods[0].invoke(instance, invocationContext);
} // else it can be a @PostContruct void pc(); which shouldn't be called from here
else
{
return invocationContext.proceed();
}
}
else
{
// otherwise we need to wrap the InvocationContext into an own temporary InvocationContext
// which handles multiple interceptor methods at a time
if (invocationContext instanceof MultiMethodInvocationContext)
{
// this happens while we recurse through the interceptors which have multiple interceptor-methods
MultiMethodInvocationContext mmInvocationContext = (MultiMethodInvocationContext) invocationContext;
int methodIndex = mmInvocationContext.getCurrentInterceptorIdx();
if (methodIndex < (interceptorMethods.length -1))
{
return interceptorMethods[methodIndex].invoke(instance, invocationContext);
}
else
{
return interceptorMethods[methodIndex].invoke(instance, mmInvocationContext.getWrapped());
}
}
else
{
// We need to create the wrapper InvocationContext on the first time.
// This will internally walk through all the methods
MultiMethodInvocationContext mmInvocationContext
= new MultiMethodInvocationContext(invocationContext, interceptionType, instance, this);
return mmInvocationContext.proceed();
}
}
}
catch (InvocationTargetException ite)
{
throw ExceptionUtil.throwAsRuntimeException(ite.getCause());
}
catch (Exception e)
{
throw ExceptionUtil.throwAsRuntimeException(e);
}
}
/**
* An InvocationContext wraper for handling multiple interceptor methods.
* We will first make sure the own interceptor methods get handled and only
* then continue with the wrapped InvocationHandler.
*/
public static class MultiMethodInvocationContext implements InvocationContext
{
private final InvocationContext wrapped;
private final Interceptor interceptor;
private final InterceptionType interceptionType;
private final Object instance;
private int currentInterceptorIdx;
public MultiMethodInvocationContext(InvocationContext wrapped,
InterceptionType interceptionType,
Object instance,
Interceptor interceptor)
{
this.wrapped = wrapped;
this.interceptor = interceptor;
this.interceptionType = interceptionType;
this.instance = instance;
currentInterceptorIdx = 0; // we start with method 0
}
public int getCurrentInterceptorIdx()
{
// in proceed we +1 to handle index properly whatever the stack is but we need to -1 here to not skip
// index 0 when we have multiple interceptors
return currentInterceptorIdx - 1;
}
public InvocationContext getWrapped()
{
return wrapped;
}
@Override
public Object proceed() throws Exception
{
currentInterceptorIdx++;
return interceptor.intercept(interceptionType, instance, this);
}
@Override
public Map getContextData()
{
return wrapped.getContextData();
}
@Override
public Method getMethod()
{
return wrapped.getMethod();
}
@Override
public Object getTarget()
{
return wrapped.getTarget();
}
@Override
public Object getTimer()
{
return wrapped.getTimer();
}
@Override
public Object[] getParameters()
{
return wrapped.getParameters();
}
@Override
public void setParameters(Object[] parameters)
{
wrapped.setParameters(parameters);
}
// @Override
public Constructor getConstructor()
{
return null;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy