io.smallrye.faulttolerance.config.FallbackConfig Maven / Gradle / Ivy
/*
* Copyright 2017 Red Hat, Inc, and 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 io.smallrye.faulttolerance.config;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.security.PrivilegedActionException;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import javax.enterprise.inject.spi.AnnotatedMethod;
import org.eclipse.microprofile.faulttolerance.Fallback;
import org.eclipse.microprofile.faulttolerance.FallbackHandler;
import org.eclipse.microprofile.faulttolerance.exceptions.FaultToleranceDefinitionException;
/**
* @author Antoine Sabot-Durand
*/
public class FallbackConfig extends GenericConfig {
public static final String VALUE = "value";
public static final String FALLBACK_METHOD = "fallbackMethod";
public static final String APPLY_ON = "applyOn";
public static final String SKIP_ON = "skipOn";
public FallbackConfig(Class> beanClass, Method method) {
super(Fallback.class, beanClass, method);
}
public FallbackConfig(AnnotatedMethod> annotatedMethod) {
super(Fallback.class, annotatedMethod);
}
@Override
public void validate() {
if (!"".equals(get(FALLBACK_METHOD))) {
if (!Fallback.DEFAULT.class.equals(get(VALUE))) {
throw new FaultToleranceDefinitionException(INVALID_FALLBACK_ON + getMethodInfo()
+ " : fallback handler class and fallback method can't be specified both at the same time");
}
Method fallbackMethod;
try {
fallbackMethod = SecurityActions.getDeclaredMethod(
beanClass,
method.getDeclaringClass(),
get(FALLBACK_METHOD),
method.getGenericParameterTypes());
} catch (PrivilegedActionException e) {
throw new FaultToleranceDefinitionException(INVALID_FALLBACK_ON + getMethodInfo()
+ " : can't find fallback method " + get(FALLBACK_METHOD) + " with same parameters as the method", e);
}
if (fallbackMethod == null) {
throw new FaultToleranceDefinitionException(INVALID_FALLBACK_ON + getMethodInfo()
+ " : can't find fallback method " + get(FALLBACK_METHOD) + " with same parameters as the method");
}
if (!method.getReturnType().equals(void.class)
&& !isAssignableFrom(method.getGenericReturnType(), fallbackMethod.getGenericReturnType())) {
throw new FaultToleranceDefinitionException(INVALID_FALLBACK_ON + getMethodInfo() + " : fallback method "
+ get(FALLBACK_METHOD) + " must have a return type assignable to method's return type");
}
}
if (!Fallback.DEFAULT.class.equals(get(VALUE))) {
Class> fbhc = get(VALUE);
Type fallbackType = null;
for (Type genericInterface : fbhc.getGenericInterfaces()) {
if (genericInterface instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) genericInterface;
if (parameterizedType.getRawType().equals(FallbackHandler.class)) {
fallbackType = parameterizedType.getActualTypeArguments()[0];
break;
}
}
}
if (fallbackType == null || !method.getGenericReturnType().equals(fallbackType)) {
throw new FaultToleranceDefinitionException(INVALID_FALLBACK_ON + getMethodInfo()
+ " : fallback handler's type " + fallbackType + " is not the same as method's return type");
}
}
}
@Override
protected Map> getKeysToType() {
return keys2Type;
}
private static final String INVALID_FALLBACK_ON = "Invalid @Fallback on ";
private static Map> keys2Type = initKeys();
private static Map> initKeys() {
Map> keys = new HashMap<>();
keys.put(VALUE, Class.class);
keys.put(FALLBACK_METHOD, String.class);
keys.put(APPLY_ON, Class[].class);
keys.put(SKIP_ON, Class[].class);
return Collections.unmodifiableMap(keys);
}
/**
* The assignability checks are incomplete and need revision.
*
* @param type1
* @param type2
* @return {@code true} if type1 is assignable from type2
*/
private static boolean isAssignableFrom(Type type1, Type type2) {
if (type1 instanceof Class>) {
if (type2 instanceof Class>) {
return isAssignableFrom((Class>) type1, (Class>) type2);
}
if (type2 instanceof ParameterizedType) {
return isAssignableFrom((Class>) type1, (ParameterizedType) type2);
}
throw new IllegalArgumentException("Unsupported type " + type2);
}
if (type1 instanceof ParameterizedType) {
if (type2 instanceof ParameterizedType) {
return isAssignableFrom((ParameterizedType) type1, (ParameterizedType) type2);
}
throw new IllegalArgumentException("Unsupported type " + type2);
}
throw new IllegalArgumentException("Unsupported type " + type1);
}
private static boolean isAssignableFrom(Class> type1, Class> type2) {
return type1.isAssignableFrom(type2);
}
private static boolean isAssignableFrom(ParameterizedType type1, ParameterizedType type2) {
final Class> rawType1 = (Class>) type1.getRawType();
final Class> rawType2 = (Class>) type2.getRawType();
if (!rawType1.equals(rawType2)) {
return false;
}
final Type[] types1 = type1.getActualTypeArguments();
final Type[] types2 = type2.getActualTypeArguments();
if (types1.length != types2.length) {
return false;
}
for (int i = 0; i < type1.getActualTypeArguments().length; i++) {
if (!isAssignableFrom(types1[i], types2[i])) {
return false;
}
}
return true;
}
}