org.jboss.arquillian.graphene.intercept.InterceptorBuilder Maven / Gradle / Ivy
The newest version!
/**
* JBoss, Home of Professional Open Source
* Copyright 2013, Red Hat, Inc. and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.arquillian.graphene.intercept;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import net.bytebuddy.implementation.bind.annotation.Empty;
import net.bytebuddy.implementation.bind.annotation.FieldValue;
import net.bytebuddy.implementation.bind.annotation.Origin;
import net.bytebuddy.implementation.bind.annotation.RuntimeType;
import org.jboss.arquillian.graphene.bytebuddy.MethodInterceptor;
import org.jboss.arquillian.graphene.context.GrapheneContext;
import org.jboss.arquillian.graphene.proxy.Interceptor;
import org.jboss.arquillian.graphene.proxy.InvocationContext;
/**
* The builder for new interceptors for type-safe definition of intercepted methods.
*
* @author Lukas Fryc
*/
public class InterceptorBuilder {
private Map> interceptors = new HashMap<>();
public static class InterceptorImpl implements MethodInterceptor {
private final InterceptorBuilder builder;
private final Interceptor interceptor;
public InterceptorImpl(InterceptorBuilder builder, Interceptor interceptor) {
this.builder = builder;
this.interceptor = interceptor;
}
@RuntimeType
public static Object intercept(@Origin Method method,
@FieldValue("__interceptor") MethodInterceptor interceptor,
@Empty Object defaultValue) throws Throwable {
InterceptorImpl impl = (InterceptorImpl) interceptor;
impl.builder.registerMethodInterceptor(method, impl.interceptor);
return defaultValue;
}
}
public InterceptorBuilder() {
}
/**
* Returns proxy for given type which execution causes remembering what method should be intercepted by given interceptor.
*
* @param type the type of the generated proxy
* @param interceptor the interceptor for intercepting methods executed on generated proxy
* @return proxy for executing methods which shoyld be intercepted by given interceptor
*/
public T interceptInvocation(Class type, final Interceptor interceptor) {
return (T) ClassImposterizer.INSTANCE.imposterise(new InterceptorImpl(this, interceptor), type);
}
private void registerMethodInterceptor(Method method, Interceptor interceptor) {
List list = this.interceptors.get(method);
if (list == null) {
list = new LinkedList<>();
this.interceptors.put(method, list);
}
list.add(interceptor);
}
/**
* Builds the final version of {@link Interceptor} intercepting all methods recorded via
* {@link #interceptInvocation(Class, Interceptor)}.
*
* @return the final version of {@link Interceptor}
*/
public Interceptor build() {
return new Interceptor() {
final Map> interceptors = new HashMap>(
InterceptorBuilder.this.interceptors);
@Override
public Object intercept(final InvocationContext originalContext) throws Throwable {
Collection interceptors = this.interceptors.get(originalContext.getMethod());
if (interceptors != null) {
final Iterator iterator = interceptors.iterator();
if (iterator.hasNext()) {
Interceptor interceptor = iterator.next();
return interceptor.intercept(new InvocationContext() {
@Override
public Object invoke() throws Throwable {
if (iterator.hasNext()) {
return iterator.next().intercept(this);
} else {
return originalContext.invoke();
}
}
@Override
public Object getTarget() {
return originalContext.getTarget();
}
@Override
public Object getProxy() {
return originalContext.getProxy();
}
@Override
public Method getMethod() {
return originalContext.getMethod();
}
@Override
public Object[] getArguments() {
return originalContext.getArguments();
}
@Override
public GrapheneContext getGrapheneContext() {
return originalContext.getGrapheneContext();
}
});
}
}
return originalContext.invoke();
}
@Override
public int getPrecedence() {
return 0;
}
};
}
}