Feign Reflection ErrorDecoder
This small library implements ErrorDecoder
to provide a simple way to map a key returned on an API to a specific exception declared thrown on the client interface.
It's very useful in an application with many microservices calling each others when you want to handle specific checked or runtime exceptions on the client side.
Blog post explaining more about the rationale of this library.
The plugin is available on Maven Central :
<dependency>
<groupId>com.coveo</groupId>
<artifactId>feign-error-decoder</artifactId>
<version>3.1.0</version>
</dependency>
A complete example is shown in this repo as the base setup for the unit tests.
11+
In order to use this library, you need a base exception which all the exceptions declared thrown on the client interface will inherit from. This exception needs to provide a way to access a unique String
key per subclass.
public abstract class ServiceException extends Exception {
private String errorCode;
//Constructors omitted
public String getErrorCode() {
return errorCode;
}
}
A Class representing the body of the response sent on the API in case of error is also needed. This is the Class that will be instantiated by Feign Decoder
. It needs a method to get the key that will be used to get the correct exception. It can also optionally have a message that will be injected in the detailMessage
field of Throwable
.
public class ErrorCodeAndMessage {
private String message;
private String errorCode;
//getters and setters omitted
With these requirements met, all left to do is to extend ReflectionErrorDecoder
and implement the abstract methods required like this :
import feign.Response;
import feign.codec.ErrorDecoder;
public class ServiceExceptionErrorDecoder
extends ReflectionErrorDecoder<ErrorCodeAndMessage, ServiceException> {
public ServiceExceptionErrorDecoder(Class<?> apiClass) {
super(apiClass, ErrorCodeAndMessage.class, ServiceException.class);
}
@Override
protected String getKeyFromException(ServiceException exception) {
return exception.getErrorCode();
}
@Override
protected String getKeyFromResponse(ErrorCodeAndMessage apiResponse) {
return apiResponse.getErrorCode();
}
@Override
protected String getMessageFromResponse(ErrorCodeAndMessage apiResponse) {
return apiResponse.getMessage();
}
}
Use the Feign builder to inject the ErrorDecoder
in your client.
The supported annotations on the interfaces are Feign's @RequestLine
and Spring @RequestMapping
. As of version 1.2.0, it also supports @GetMapping
, @PostMapping
, @PutMapping
, @DeleteMapping
and @PatchMapping
.
In versions 1.x, this library sets the detailMessage
field of the Throwable
instance via reflection using the message from the ReflectionErrorDecoder::getMessageFromResponse
method. This is not supported anymore in JDK 16+ as it's considered an illegal reflective access.
To work around this, a new interface ExceptionMessageSetter
has been introduced. It is meant to be used on the base exception class like this :
public abstract class ServiceException extends Exception implements ExceptionMessageSetter {
private String detailMessage;
@Override
public String getMessage() {
return detailMessage == null ? super.getMessage() : detailMessage;
}
@Override
public void setExceptionMessage(String detailMessage) {
this.detailMessage = detailMessage;
}
}
This way, the Throwable
message field will be gracefully set without using illegal reflective access. Having an interface instead of an abstract class was decided to make sure this library doesn't creep up an abstract class in your code. However, it has the tradeoff that you need to override the Throwable::getMessage
yourself.
All the library dependencies have been made <optional>true</optional>
so you can link on this library in your base exception package without having to transitively pull on Feign.
A ClassHierarchySupplier
interface is used to support classpath scanning to fetch the hierarchy of abstract exception classes. This allows you to declare a specific base exception as thrown on the client interface and let the interface scan all the possible exceptions that can be thrown.
An optional dependency on Spring Context is included in the library to enable this. All you need to do is have Spring framework available in your project and the proper implementation will be instantiated. By default, it will scan the exception children in all packages. To restrict the base package to be scanned, simply use the constructor with the basePackage
field.
A default implementation is not provided at the moment. Feel free to submit a PR if you implement it!
By default, this project uses the JacksonDecoder
implementation of Feign Decoder
interface. A protected setter is available to use your own Decoder
.
ErrorDecoder.Default
is used by default when no exception is found in the scanned exceptions. A protected setter is available to use your own fallback ErrorDecoder
.
The library has a default list of supported argument types for the exception constructors. It supports empty and constructors with any number of String
or Throwable
in any order. To extend supported exception types, just override the method protected List<Object> getSupportedConstructorArgumentInstances()
. Just make sure to return the default types of String
and Throwable
if you still want them to be supported.
PR are always welcome and please open an issue if you find any bugs or wish to request an additional feature.