OkHttp实现全局过期token自动刷新
#遇到问题:
当前开发的 App 遇到一个问题:
当请求某个接口时,由于 token 已经失效,所以接口会报错。
但是产品经理希望 app 能够马上刷新 token ,然后重复请求刚才那个接口,这个过程对用户来说是无感的。
>
也就是静默自动登录,然后继续请求:
>
请求 A 接口-》服务器返回 token 过期-》请求 token 刷新接口-》请求 A 接口
>
要实现上述需求的话,大家会如何实现呢?
#解决方案:
思路:
1.通过拦截器,获取返回的数据
2.判断token是否过期
3.如果token过期则刷新token
4.使用最新的token,重新请求网络数据
1 /** 2 * 全局自动刷新Token的拦截器 3 */ 4 public class TokenInterceptor implements Interceptor { 5 6 @Override 7 public Response intercept(Chain chain) throws IOException { 8 Request request = chain.request(); 9 Response response = chain.proceed(request); 10 LogUtil.print("response.code=" + response.code()); 11 12 if (isTokenExpired(response)) {//根据和服务端的约定判断token过期 13 LogUtil.print("静默自动刷新Token,然后重新请求数据"); 14 //同步请求方式,获取最新的Token 15 String newSession = getNewToken(); 16 //使用新的Token,创建新的请求 17 Request newRequest = chain.request() 18 .newBuilder() 19 .header("Cookie", "JSESSIONID=" + newSession) 20 .build(); 21 //重新请求 22 return chain.proceed(newRequest); 23 } 24 return response; 25 } 26 27 /** 28 * 根据Response,判断Token是否失效 29 * 30 * @param response 31 * @return 32 */ 33 private boolean isTokenExpired(Response response) { 34 if (response.code() == 404) { 35 return true; 36 } 37 return false; 38 } 39 40 /** 41 * 同步请求方式,获取最新的Token 42 * 43 * @return 44 */ 45 private String getNewToken() throws IOException { 46 // 通过一个特定的接口获取新的token,此处要用到同步的retrofit请求 47 Response_Login loginInfo = CacheManager.restoreLoginInfo(BaseApplication.getContext()); 48 String username = loginInfo.getUserName(); 49 String password = loginInfo.getPassword(); 50 51 LogUtil.print("loginInfo=" + loginInfo.toString()); 52 Call<Response_Login> call = WebHelper.getSyncInterface().synclogin(new Request_Login(username, password)); 53 loginInfo = call.execute().body(); 54 LogUtil.print("loginInfo=" + loginInfo.toString()); 55 56 loginInfo.setPassword(password); 57 CacheManager.saveLoginInfo(loginInfo); 58 return loginInfo.getSession(); 59 } 60 }
然后配置下OkHttp
HttpLoggingInterceptor logging = new HttpLoggingInterceptor(); logging.setLevel(HttpLoggingInterceptor.Level.BODY); OkHttpClient client = new OkHttpClient.Builder() .connectTimeout(15, TimeUnit.SECONDS) .readTimeout(300, TimeUnit.SECONDS) .writeTimeout(300, TimeUnit.SECONDS) .cache(new Cache(FileConstants.HTTP_CACHE_DIR, FileConstants.CACHE_SIZE)) .addInterceptor(interceptor) // .addInterceptor(new MockInterceptor()) .addInterceptor(new TokenInterceptor()) // .addInterceptor(new RetryIntercepter(3)) .addInterceptor(logging) .build();