 
                        
        
                        
        org.jboss.weld.interceptor.proxy.AroundInvokeInvocationContext Maven / Gradle / Ivy
                 Go to download
                
        
                    Show more of this group  Show more artifacts with this name
Show all versions of weld-servlet-shaded Show documentation
                Show all versions of weld-servlet-shaded Show documentation
This jar bundles all the bits of Weld and CDI required for running in a Servlet container.
                
             The newest version!
        
        /*
 * JBoss, Home of Professional Open Source
 * Copyright 2014, Red Hat, Inc., and individual contributors
 * by the @authors tag. See the copyright.txt in the distribution for a
 * full listing of individual contributors.
 *
 * 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.jboss.weld.interceptor.proxy;
import static org.jboss.weld.util.reflection.Reflections.unwrapInvocationTargetException;
import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.List;
import java.util.Map;
import java.util.Set;
import jakarta.interceptor.InvocationContext;
import org.jboss.weld.bean.proxy.CombinedInterceptorAndDecoratorStackMethodHandler;
import org.jboss.weld.bean.proxy.InterceptionDecorationContext;
import org.jboss.weld.bean.proxy.InterceptionDecorationContext.Stack;
/**
 * For AroundInvoke interception type we use a special type of InvocationContext. Unlike the default one, this one is does not
 * track
 * the position of the interception in a mutable field but is instead immutable.
 *
 * This allows:
 * 
 * - interception to be repeatable - e.g. an interceptor may call InvocationContext.proceed() multiple times*
- interception to continue in a different thread - implementing {@link jakarta.ejb.Asynchronous} with interceptors*
*
 * This however also requires that for each interceptor in the chain we create a new instance of
 * {@link AroundInvokeInvocationContext}.
 * Context data and method parameters are mutable. We do not guard them anyhow - the expectation for them is to be effectively
 * immutable
 * by only being modified before or after dispatch. We also assume that the dispatch safely propagates the state of
 * {@link InvocationContext}
 * from one thread to the other.
 *
 * @author Jozef Hartinger
 * @see TerminalAroundInvokeInvocationContext
 * @see NonTerminalAroundInvokeInvocationContext
 *
 */
abstract class AroundInvokeInvocationContext extends AbstractInvocationContext {
    public static AroundInvokeInvocationContext create(Object instance, Method method, Method proceed, Object[] args,
            List chain,
            Set interceptorBindings, Stack stack) {
        CombinedInterceptorAndDecoratorStackMethodHandler currentHandler = (stack == null) ? null : stack.peek();
        if (chain.size() == 1) {
            return new TerminalAroundInvokeInvocationContext(instance, method, proceed, args, null, interceptorBindings,
                    currentHandler);
        } else {
            return new NonTerminalAroundInvokeInvocationContext(instance, method, proceed, args, interceptorBindings, chain,
                    currentHandler);
        }
    }
    final CombinedInterceptorAndDecoratorStackMethodHandler currentHandler;
    AroundInvokeInvocationContext(Object target, Method method, Method proceed, Object[] parameters,
            Map contextData,
            Set interceptorBindings, CombinedInterceptorAndDecoratorStackMethodHandler currentHandler) {
        super(target, method, proceed, parameters, contextData, interceptorBindings);
        this.currentHandler = currentHandler;
    }
    @Override
    public Object proceed() throws Exception {
        final Stack stack = InterceptionDecorationContext.startIfNotOnTop(currentHandler);
        try {
            return proceedInternal();
        } catch (InvocationTargetException e) {
            throw unwrapInvocationTargetException(e);
        } finally {
            if (stack != null) {
                stack.end();
            }
        }
    }
    abstract Object proceedInternal() throws Exception;
}
    © 2015 - 2025 Weber Informatics LLC | Privacy Policy