實戰!退出登錄時如何借助外力使JWT令牌失效?


大家好,我是不才陳某~

今天這篇文章介紹一下如何在修改密碼修改權限注銷等場景下使JWT失效。

文章的目錄如下:

解決方案

JWT最大的一個優勢在於它是無狀態的,自身包含了認證鑒權所需要的所有信息,服務器端無需對其存儲,從而給服務器減少了存儲開銷。

但是無狀態引出的問題也是可想而知的,它無法作廢未過期的JWT。舉例說明注銷場景下,就傳統的cookie/session認證機制,只需要把存在服務器端的session刪掉就OK了。

但是JWT呢,它是不存在服務器端的啊,好的那我刪存在客戶端的JWT行了吧。額,社會本就復雜別再欺騙自己了好么,被你在客戶端刪掉的JWT還是可以通過服務器端認證的。

使用JWT要非常明確的一點:JWT失效的唯一途徑就是等待時間過期

但是可以借助外力保存JWT的狀態,這時就有人問了:你這不是打臉嗎?用JWT就因為它的無狀態性,這時候又要保存它的狀態?

其實不然,這不被逼上梁山了嗎?不使用外力保存JWT的狀態,你說如何實現注銷失效?

常用的方案有兩種,白名單黑名單方式。

1、白名單

白名單的邏輯很簡單:認證通過時,將JWT存入redis中,注銷時,將JWT從redis中移出。這種方式和cookie/session的方式大同小異。

2、黑名單

黑名單的邏輯也非常簡單:注銷時,將JWT放入redis中,並且設置過期時間為JWT的過期時間;請求資源時判斷該JWT是否在redis中,如果存在則拒絕訪問。

白名單和黑名單這兩種方案都比較好實現,但是黑名單帶給服務器的壓力遠遠小於白名單,畢竟注銷不是經常性操作。

黑名單方式實現

下面以黑名單的方式介紹一下如何在網關層面實現JWT的注銷失效。

究竟向Redis中存儲什么?

如果直接存儲JWT令牌可行嗎?當然可行,不過JWT令牌可是很長的哦,這樣對內存的要求也是挺高的。

熟悉JWT令牌的都知道,JWT令牌中有一個jti字段,這個字段可以說是JWT令牌的唯一ID了,如下:

因此可以將這個jti字段存入redis中,作為唯一令牌標識,這樣一來是不是節省了很多的內存?

如何實現呢? 分為兩步:

  1. 網關層的全局過濾器中需要判斷黑名單是否存在當前JWT
  2. 注銷接口中將JWT的jti字段作為key存放到redis中,且設置了JWT的過期時間

1、網關層解析JWT的jti、過期時間放入請求頭中

在網關的全局過濾器GlobalAuthenticationFilter中直接從令牌中解析出jti過期時間

這里的邏輯分為如下步驟:

  1. 解析JWT令牌的jti和過期時間
  2. 根據jti從redis中查詢是否存在黑名單中,如果存在則直接攔截,否則放行
  3. 將解析的jti和過期時間封裝到JSON中,傳遞給下游微服務

關鍵代碼如下:

2、下游微服務的過濾器修改

還記得上篇文章:實戰干貨!Spring Cloud Gateway 整合 OAuth2.0 實現分布式統一認證授權!中微服務的過濾器AuthenticationFilter嗎?

AuthenticationFilter這個過濾器用來解密網關層傳遞的JSON數據,並將其封裝到Request中,這樣在業務方法中便可以隨時獲取到想要的用戶信息。

這里我是把JWT相關的信息同時封裝到了Request中,實體類為JwtInformation,如下:

LoginVal繼承了JwtInformation,如下:

此時AuthenticationFilter這個過濾器修改起來就很簡單了,只需要將jti和過期時間封裝到LoginVal中即可,關鍵代碼如下:

邏輯很簡單,上圖都有標注。

3、注銷接口實現

之前文章中並沒有提供注銷接口,因為無狀態的JWT根本不需要退出登錄,傻等着過期唄。

當然為了實現注銷登錄,借助了Redis,那么注銷接口必不可少了。

邏輯很簡單,直接將退出登錄的JWT令牌的jti設置到Redis中,過期時間設置為JWT過期時間即可。代碼如下:

OK了,至此已經實現了JWT注銷登錄的功能.......

涉及到的三個模塊的改動,分別如下:

名稱 功能
oauth2-cloud-auth-server OAuth2.0認證授權服
oauth2-cloud-gateway 網關服務
oauth2-cloud-auth-common 公共模塊

總結

思想很簡單,JWT既然是無狀態的,只能借助Redis記錄它的狀態,這樣才能達到使其失效的目的。

測試

業務基本完成了,下面走一個流程測試一下,如下:

1、登錄,申請令牌

2、拿着令牌訪問接口

該令牌並沒有注銷,因此可以正常訪問,如下:

3、調用接口注銷登錄

請求如下:

4、拿着注銷的令牌訪問接口

由於令牌已經注銷了,因此肯定訪問不通接口,返回如下:


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM