伪客户端重试器具有新的请求拦截器?

时间:2018-11-27 13:24:58

标签: java interceptor spring-cloud-feign feign

我目前正在手动构建伪装客户端,并将拦截器传递给该客户端以进行授权。我想要一个更智能的Retryer来处理某些响应代码。

public class myErrorEncoder extends ErrorDecoder.Default {

@Override
public Exception decode(final String methodKey, final Response response) {
    if (response.status() == 401) {
        String token = refreshToken();  // I would like to refresh the token and Edit the client
        return new RetryableException("Token Expired will retry it", null);
    } else {
        return super.decode(methodKey, response);
    }
}

}

拦截器

@Bean public CustomInterceptor getInterceptor(String token) {
    return new CustomInterceptor(token);}

伪装者

 private <T> T feignBuild(final Class<T> clazz, final String uri, final String token) {
    return Feign
            .builder().client(new ApacheHttpClient())
            .encoder(new GsonEncoder())
            .decoder(new ResponseEntityDecoder(feignDecoder())
            .retryer(new Retryer.Default(1,100,3))
            .errorDecoder(new ErrorDecoder())
            .requestInterceptor(getInterceptor(token))
            .contract(new ClientContract())
            .logger(new Slf4jLogger(clazz)).target(clazz, uri);
}

现在,我想使用刷新的令牌更新虚拟客户端并重试。 有没有一种方法可以访问客户端实例并对其进行配置。

1 个答案:

答案 0 :(得分:0)

您对拦截器的使用不正确。拦截器在重试期间被重新应用,但是它们仅被实例化一次,并且预期是线程安全的。要实现您想要的目标,需要将令牌生成与拦截器分开,并让拦截器请求新令牌。

public class TokenInterceptor() {
   TokenService tokenService;

   public TokenInterceptor(TokenService tokenService) {
      this.tokenService = tokenService;
   }

   public void apply(RequestTemplate template) {
      /* getToken() should create a new token */
      String token = this.tokenService.getToken();
      template.header("Authorization", "Bearer " + token);
   }
}

这将确保每个重试周期都创建一个新令牌。