OkHttp實現全局過時token自動刷新

#遇到問題:

當前開發的 App 遇到一個問題:服務器

當請求某個接口時,因爲 token 已經失效,因此接口會報錯。
可是產品經理但願 app 可以立刻刷新 token ,而後重複請求剛纔那個接口,這個過程對用戶來講是無感的。
>
也就是靜默自動登陸,而後繼續請求:
>
請求 A 接口-》服務器返回 token 過時-》請求 token 刷新接口-》請求 A 接口
>
要實現上述需求的話,你們會如何實現呢?網絡

#解決方案:

思路:
1.經過攔截器,獲取返回的數據
2.判斷token是否過時
3.若是token過時則刷新token
4.使用最新的token,從新請求網絡數據app

 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 }

而後配置下OkHttpide

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();
相關文章
相關標籤/搜索