All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.micronaut.aop.chain.MethodInterceptorChain Maven / Gradle / Ivy

/*
 * Copyright 2017-2020 original authors
 *
 * 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
 *
 * https://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 io.micronaut.aop.chain;

import io.micronaut.aop.Interceptor;
import io.micronaut.aop.InterceptorKind;
import io.micronaut.aop.InterceptorRegistry;
import io.micronaut.aop.Introduced;
import io.micronaut.aop.MethodInterceptor;
import io.micronaut.aop.MethodInvocationContext;
import io.micronaut.aop.exceptions.UnimplementedAdviceException;
import io.micronaut.context.BeanContext;
import io.micronaut.context.BeanRegistration;
import io.micronaut.context.BeanResolutionContext;
import io.micronaut.context.DefaultBeanContext;
import io.micronaut.core.annotation.AnnotationMetadata;
import io.micronaut.core.annotation.AnnotationValue;
import io.micronaut.core.annotation.Internal;
import io.micronaut.core.annotation.NonNull;
import io.micronaut.core.annotation.Nullable;
import io.micronaut.core.annotation.UsedByGeneratedCode;
import io.micronaut.core.type.ReturnType;
import io.micronaut.core.util.ArrayUtils;
import io.micronaut.inject.BeanDefinition;
import io.micronaut.inject.ExecutableMethod;
import io.micronaut.inject.qualifiers.Qualifiers;

import java.lang.reflect.Method;
import java.util.Collection;
import java.util.Objects;

import static io.micronaut.core.util.ArrayUtils.EMPTY_OBJECT_ARRAY;

/**
 * An internal representation of the {@link Interceptor} chain. This class implements {@link MethodInvocationContext} and is
 * consumed by the framework itself and should not be used directly in application code.
 *
 * @param  type
 * @param  result
 * @author Graeme Rocher
 * @since 1.0
 */
@Internal
@UsedByGeneratedCode
public final class MethodInterceptorChain extends InterceptorChain implements MethodInvocationContext {

    private final @Nullable InterceptorKind kind;

    /**
     * Constructor for empty parameters.
     *
     * @param interceptors array of interceptors
     * @param target target
     * @param executionHandle executionHandle
     */
    @UsedByGeneratedCode
    public MethodInterceptorChain(Interceptor[] interceptors, T target, ExecutableMethod executionHandle) {
        this(interceptors, target, executionHandle, (InterceptorKind) null);
    }

    /**
     * Constructor for empty parameters.
     *
     * @param interceptors array of interceptors
     * @param target target
     * @param executionHandle executionHandle
     * @param kind The interception kind
     */
    public MethodInterceptorChain(
        Interceptor[] interceptors,
        T target,
        ExecutableMethod executionHandle,
        @Nullable InterceptorKind kind) {
        super(interceptors, target, executionHandle, EMPTY_OBJECT_ARRAY);
        this.kind = kind;
    }


    /**
     * Constructor.
     *
     * @param interceptors array of interceptors
     * @param target target
     * @param executionHandle executionHandle
     * @param originalParameters originalParameters
     */
    @UsedByGeneratedCode
    public MethodInterceptorChain(Interceptor[] interceptors, T target, ExecutableMethod executionHandle, Object... originalParameters) {
        super(interceptors, target, executionHandle, originalParameters);
        this.kind = null;
    }

    @Override
    @NonNull
    public InterceptorKind getKind() {
        return this.kind != null ? kind : target instanceof Introduced ? InterceptorKind.INTRODUCTION : InterceptorKind.AROUND;
    }

    @Override
    public R invoke(T instance, Object... arguments) {
        return new MethodInterceptorChain<>(interceptors, instance, executionHandle, originalParameters).proceed();
    }

    @Override
    public boolean isSuspend() {
        return executionHandle.isSuspend();
    }

    @Override
    public boolean isAbstract() {
        return executionHandle.isAbstract();
    }

    @Override
    public R proceed() throws RuntimeException {
        Interceptor interceptor;
        if (interceptorCount == 0 || index == interceptorCount) {
            if (target instanceof Introduced && executionHandle.isAbstract()) {
                throw new UnimplementedAdviceException(executionHandle);
            } else {
                return executionHandle.invoke(target, getParameterValues());
            }
        } else {
            interceptor = this.interceptors[index++];
            if (LOG.isTraceEnabled()) {
                LOG.trace("Proceeded to next interceptor [{}] in chain for method invocation: {}", interceptor, executionHandle);
            }

            if (interceptor instanceof MethodInterceptor) {
                return ((MethodInterceptor) interceptor).intercept(this);
            } else {
                return interceptor.intercept(this);
            }
        }
    }

    @Override
    public String getMethodName() {
        return executionHandle.getMethodName();
    }

    @Override
    public Class[] getArgumentTypes() {
        return executionHandle.getArgumentTypes();
    }

    @Override
    public Method getTargetMethod() {
        return executionHandle.getTargetMethod();
    }

    @Override
    public ReturnType getReturnType() {
        return executionHandle.getReturnType();
    }

    @NonNull
    @Override
    public Class getDeclaringType() {
        return executionHandle.getDeclaringType();
    }

    @Override
    public String toString() {
        return executionHandle.toString();
    }

    @NonNull
    @Override
    public ExecutableMethod getExecutableMethod() {
        return executionHandle;
    }

    /**
     * Internal method that handles the logic for executing {@link InterceptorKind#POST_CONSTRUCT} interception.
     *
     * @param resolutionContext The resolution context
     * @param beanContext The bean context
     * @param definition The definition
     * @param postConstructMethod The post construct method
     * @param bean The bean
     * @param  The bean type
     * @return the bean instance
     * @since 3.0.0
     */
    @Internal
    @UsedByGeneratedCode
    @NonNull
    public static  T1 initialize(
        @NonNull BeanResolutionContext resolutionContext,
        @NonNull BeanContext beanContext,
        @NonNull BeanDefinition definition,
        @NonNull ExecutableMethod postConstructMethod,
        @NonNull T1 bean) {
        return doIntercept(
            resolutionContext,
            beanContext,
            definition,
            postConstructMethod,
            bean,
            InterceptorKind.POST_CONSTRUCT
        );
    }

    /**
     * Internal method that handles the logic for executing {@link InterceptorKind#PRE_DESTROY} interception.
     *
     * @param resolutionContext The resolution context
     * @param beanContext The bean context
     * @param definition The definition
     * @param preDestroyMethod The pre destroy method
     * @param bean The bean
     * @param  The bean type
     * @return the bean instance
     * @since 3.0.0
     */
    @Internal
    @UsedByGeneratedCode
    @NonNull
    public static  T1 dispose(
        @NonNull BeanResolutionContext resolutionContext,
        @NonNull BeanContext beanContext,
        @NonNull BeanDefinition definition,
        @NonNull ExecutableMethod preDestroyMethod,
        @NonNull T1 bean) {
        return doIntercept(
            resolutionContext,
            beanContext,
            definition,
            preDestroyMethod,
            bean,
            InterceptorKind.PRE_DESTROY
        );
    }

    @SuppressWarnings({"unchecked", "rawtypes"})
    private static  T1 doIntercept(
        BeanResolutionContext resolutionContext,
        BeanContext beanContext,
        BeanDefinition definition,
        ExecutableMethod interceptedMethod,
        T1 bean,
        InterceptorKind kind) {
        final AnnotationMetadata annotationMetadata = interceptedMethod.getAnnotationMetadata();
        final Collection> binding = resolveInterceptorValues(annotationMetadata, kind);

        final Collection>> resolved = ((DefaultBeanContext) beanContext).getBeanRegistrations(
            resolutionContext,
            Interceptor.ARGUMENT,
            Qualifiers.byInterceptorBindingValues(binding)
        );
        final InterceptorRegistry interceptorRegistry = beanContext.getBean(InterceptorRegistry.ARGUMENT);
        final Interceptor[] resolvedInterceptors = interceptorRegistry
            .resolveInterceptors(
                (ExecutableMethod) interceptedMethod,
                (Collection) resolved,
                kind
            );

        if (ArrayUtils.isNotEmpty(resolvedInterceptors)) {
            final MethodInterceptorChain chain = new MethodInterceptorChain<>(
                resolvedInterceptors,
                bean,
                interceptedMethod,
                kind
            );
            return Objects.requireNonNull(
                chain.proceed(),
                kind.name() + " interceptor chain illegal returned null for type: " + definition.getBeanType()
            );
        } else {
            return interceptedMethod.invoke(bean);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy