在使用feign進行遠端方法呼叫時,如果遠端服務端方法出現異常,客戶端有時需要捕獲,並且把異常資訊返回給前端,而如果在開啟熔斷
之後,這個異常會被消化,所以說,如果希望拿到服務端異常,feign.hystrix.enable
需要設定為false,而當不開熔斷時,我們也有幾種方法把拿到服務端的異常資訊,下面總結一下。
- try...catch方法
- feign異常攔截器
try...catch方法
這個方法比較直觀,也最好理解,但業務層會有很多try...catch這種程式碼,所以不是很推薦。
注意:雖然服務端方法返回為ResponseEntity,但出現4xx,5xx這些異常時,還是會丟擲一個FeignException的異常,而這對於服務來說,向前端丟擲的還是5xx的服務端異常,這不是我們希望看到的。
try {
ResponseEntity<?> body = productClient.add(name);
if (body.getStatusCode().equals(HttpStatus.OK)) {
return "test";
} else {
return body.getBody().toString();
}
} catch (FeignException ex) {
return ex.getMessage();
}
feign異常攔截器
我們比較推薦使用這個方法,註冊一個bean物件,當feign呼叫出現異常時,會觸發這個方法,然後我們再統一處理這個異常,向下流繼續丟擲加工後的錯誤資訊。
/**
* feign異常攔截器,當從feign丟擲異常時走這個物件.
*/
@Configuration
@Slf4j
public class FeignClientErrorDecoder implements ErrorDecoder {
@Override
public Exception decode(String methodKey, Response response) {
logger.info("feign client response:", response);
String body = null;
try {
body = Util.toString(response.body().asReader());
} catch (IOException e) {
logger.error("feign.IOException", e);
}
if (response.status() >= 400 && response.status() <= 500) {
throw Exceptions.badRequestParams(body);
}
return errorStatus(methodKey, response);
}
}
注意,使用這個方式,需要在熔斷器關閉時才起作用,因為它們的執行過程是,先走這個攔截器,再走熔斷的fallback,所以這個異常會被熔斷吞掉,返回狀態為200,返回值為你的fallback的預設值。