我目前正在手动构建伪装客户端,并将拦截器传递给该客户端以进行授权。我想要一个更智能的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);
}
现在,我想使用刷新的令牌更新虚拟客户端并重试。 有没有一种方法可以访问客户端实例并对其进行配置。
答案 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);
}
}
这将确保每个重试周期都创建一个新令牌。