Feign Client 超時時間配置不生效的解決

Feign Client 超時時間配置不生效

解決方案

Feign Client 的 connectTimeout 和 readTimeout 需要同時配置:

feign.client.config.my-api.connectTimeout = 10000
feign.client.config.my-api.readTimeout = 600000

如果隻配置瞭 readTimeout 或 connectTimeout 的其中一條,配置不生效。

問題描述

Feign Client 配置如下:

feign.client.config.default.connectTimeout = 3000
feign.client.config.default.readTimeout = 3000
feign.client.config.my-api.readTimeout = 600000

默認超時時間配置瞭 3 秒。

由於 my-api 的響應數據較大,於是配置瞭 10 分鐘超時時間。

執行後發現,my-api 請求仍然 3 秒後就拋出如下異常:

Caused by: feign.RetryableException: Read timed out executing POST http://my-api
 at feign.FeignException.errorExecuting(FeignException.java:67)
 at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java:104)
 at feign.SynchronousMethodHandler.invoke(SynchronousMethodHandler.java:76)
 at feign.hystrix.HystrixInvocationHandler$1.run(HystrixInvocationHandler.java:108)
 at com.netflix.hystrix.HystrixCommand$2.call(HystrixCommand.java:302)
 at com.netflix.hystrix.HystrixCommand$2.call(HystrixCommand.java:298)
 at rx.internal.operators.OnSubscribeDefer.call(OnSubscribeDefer.java:46)

Feign Client的各種超時時間設置

在Spring Cloud微服務架構中,大部分公司都是利用Open Feign進行服務間的調用,而比較簡單的業務使用默認配置是不會有多大問題的,但是如果是業務比較復雜,服務要進行比較繁雜的業務計算,那後臺很有可能會出現Read Timeout這個異常,因此定制化配置超時時間就有必要瞭。

1. Feign Client Configuration

# 默認開啟
feign.httpclient.enabled=false
# 默認關閉
feign.okhttp.enabled=true
# 默認關閉
feign.hystrix.enabled=false
# 默認關閉
feign.sentinel.enabled=true
# default context 連接超時時間
feign.client.config.default.connectTimeout = 5000
# default context 讀超時時間
feign.client.config.default.readTimeout = 10000
# 設置重試處理器,默認直接拋出異常
# feign.client.config.default.retryer = Class<Retryer>
# 設置日志級別,默認NONE
# feign.client.config.default.loggerLevel = FULL

2. Hystrix Configuration

# 全局設置超時:
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds: 30000

hystrix在ribbon的外層處理。

3. Ribbon Configuration

# 連接超時時間,默認為1秒,該值會被FeignClient配置connectTimeout覆蓋
ribbon.ConnectTimeout=5000
# 讀超時時間,默認為1秒,該值會被FeignClient配置readTimeout覆蓋
ribbon.ReadTimeout=5000
# 最大重試次數
ribbon.MaxAutoRetries=1

當Ribbon調用接口發送連接異常或者超時異常時會觸發Ribbon 的重試機制。

4. OkHttp Client Configuration

所設置的連接時間和超時時間最後會動態設置到OkHttpClient中,最底層也就是Socket的連接時間和讀超時時間。也就是說,直接配置OkHttpClient是無效的。

解決方案:添加OkHttp Client的請求Interceptor,動態設置超時時間。

@Bean("okHttpClient")
public OkHttpClient okHttpClient(ConnectionPool connectionPool) {
 return new OkHttpClient().newBuilder().connectionPool(connectionPool)
             // 改值在FeignClient體系中會被動態覆蓋
   .connectTimeout(6, TimeUnit.SECONDS)
             // 改值在FeignClient體系中會被動態覆蓋
   .readTimeout(VmcConstants.TEN_SECONDS, TimeUnit.SECONDS)
             // 添加攔截器,支持動態設置超時時間
   .addInterceptor(new OkHttpClientDynamicTimeoutInterceptor())
            .eventListener(eventListener())
   .build();
}

5. 小結一下吧

1.如何配置好Hystrix和Ribbon的超時時間呢?

其實是有套路的,因為Feign的請求:其實是Hystrix+Ribbon。Hystrix在最外層,然後再到Ribbon,最後裡面的是http請求。所以說。Hystrix的熔斷時間必須大於Ribbon的 ( ConnectTimeout + ReadTimeout )。而如果Ribbon開啟瞭重試機制,還需要乘以對應的重試次數,保證在Ribbon裡的請求還沒結束時,Hystrix的熔斷時間不會超時。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: