Spring Security中用JWT退出登錄時(shí)遇到的坑
最近有個(gè)粉絲提了個(gè)問題,說他在Spring Security中用JWT做退出登錄的時(shí)無法獲取當(dāng)前用戶,導(dǎo)致無法證明“我就是要退出的那個(gè)我”,業(yè)務(wù)失??!經(jīng)過我一番排查找到了原因,而且這個(gè)錯(cuò)誤包括我自己的大部分人都犯過。
Session會(huì)話
之所以要說Session會(huì)話,是因?yàn)镾pring Security默認(rèn)配置就是有會(huì)話的,所以當(dāng)你登錄以后Session就會(huì)由服務(wù)端保持直到你退出登錄。只要Session保持住,你的請(qǐng)求只要進(jìn)入服務(wù)器就可以從 ServletRequest 中獲取到當(dāng)前的 HttpSession ,然后會(huì)根據(jù) HttpSession 來加載當(dāng)前的 SecurityContext 。相關(guān)的邏輯在Spring Security默認(rèn)的過濾器 SecurityContextPersistenceFilter 中,有興趣可以看相關(guān)的源碼。
而且默認(rèn)情況下 SecurityContextPersistenceFilter 的優(yōu)先級(jí)是高于退出過濾器 LogoutFilter 的,所以能夠保證有Session會(huì)話的情況下退出一定能夠獲取當(dāng)前用戶。
無Session會(huì)話
使用了JWT后,每次請(qǐng)求都要攜帶 Bearer Token 并且被專門的過濾器攔截解析之后才能將用戶認(rèn)證信息保存到 SecurityContext 中去。參考Spring Security實(shí)戰(zhàn)干貨教程中的Token認(rèn)證實(shí)現(xiàn) JwtAuthenticationFilter ,相關(guān)邏輯為:
// 當(dāng)token匹配
if (jwtToken.equals(accessToken)) {
// 解析 權(quán)限集合 這里
JSONArray jsonArray = jsonObject.getJSONArray("roles");
List<String> roles = jsonArray.toList(String.class);
String[] roleArr = roles.toArray(new String[0]);
List<GrantedAuthority> authorities = AuthorityUtils.createAuthorityList(roleArr);
User user = new User(username, "[PROTECTED]", authorities);
// 構(gòu)建用戶認(rèn)證token
UsernamePasswordAuthenticationToken usernamePasswordAuthenticationToken = new UsernamePasswordAuthenticationToken(user, null, authorities);
usernamePasswordAuthenticationToken.setDetails(new WebAuthenticationDetailsSource().buildDetails(request));
// 放入安全上下文中
SecurityContextHolder.getContext().setAuthentication(usernamePasswordAuthenticationToken);
} else {
// token 不匹配
if (log.isDebugEnabled()){
log.debug("token : {} is not in matched", jwtToken);
}
throw new BadCredentialsException("token is not matched");
}
為什么退出登錄無法獲取當(dāng)前用戶
分析了兩種情況下用戶認(rèn)證信息的安全上下文配置后,我們回到問題的本身。來看看為什么用JWT會(huì)出現(xiàn)無法獲取當(dāng)前認(rèn)證信息的原因。在 HttpSecurity 中,那位同學(xué)是這樣配置 JwtAuthenticationFilter 的順序的:
httpSecurity.addFilterBefore(jwtAuthenticationFilter, UsernamePasswordAuthenticationFilter.class)
我們?cè)倏纯?Spring Security 過濾器排序圖:

也就說LogoutFilter執(zhí)行退出的時(shí)候,JWT還沒有被 JwtAuthenticationFilter 攔截,當(dāng)然無法獲取當(dāng)前認(rèn)證上下文 SecurityContext 。
解決方法
解決方法就是必須在 LogoutFilter 執(zhí)行前去解析JWT并將成功認(rèn)證的信息存到 SecurityContext 。我們可以這樣配置:
httpSecurity.addFilterBefore(jwtAuthenticationFilter, LogoutFilter.class)
這樣問題就解決了,你只要實(shí)現(xiàn)把當(dāng)前JWT作廢掉就退出登錄了。
到此這篇關(guān)于Spring Security中用JWT退出登錄時(shí)遇到的坑的文章就介紹到這了,更多相關(guān)Spring Security JWT退出登錄內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringSecurity+jwt+redis基于數(shù)據(jù)庫登錄認(rèn)證的實(shí)現(xiàn)
- spring security結(jié)合jwt實(shí)現(xiàn)用戶重復(fù)登錄處理
- Spring Boot 2結(jié)合Spring security + JWT實(shí)現(xiàn)微信小程序登錄
- springboot+jwt+springSecurity微信小程序授權(quán)登錄問題
- SpringBoot集成Spring Security用JWT令牌實(shí)現(xiàn)登錄和鑒權(quán)的方法
- Spring Security基于JWT實(shí)現(xiàn)SSO單點(diǎn)登錄詳解
- SpringSecurity+JWT實(shí)現(xiàn)登錄流程分析
相關(guān)文章
springboot validator枚舉值校驗(yàn)功能實(shí)現(xiàn)
這篇文章主要介紹了springboot validator枚舉值校驗(yàn)功能實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
Java中使用WebUploader插件上傳大文件單文件和多文件的方法小結(jié)
這篇文章主要介紹了Java中使用WebUploader插件上傳大文件單文件和多文件的方法小結(jié)的相關(guān)資料,需要的朋友可以參考下2016-06-06
Spring cloud alibaba之Ribbon負(fù)載均衡實(shí)現(xiàn)方案
Spring cloud Ribbon是基于Netflix Ribbon實(shí)現(xiàn)的一套客戶端的負(fù)載均衡工具,Ribbon客戶端提供一系列完善的配置,如超時(shí)、重試等,Ribbon也可以實(shí)現(xiàn)自己的負(fù)載均衡算法,感興趣的朋友跟隨小編一起看看吧2021-07-07
Java 動(dòng)態(tài)代理的多種實(shí)現(xiàn)方式
動(dòng)態(tài)代理實(shí)際上是JVM在運(yùn)行期動(dòng)態(tài)創(chuàng)建class字節(jié)碼并加載的過程。本文講述了Java 動(dòng)態(tài)代理的多種實(shí)現(xiàn)方式,感興趣的朋友可以選擇適合自己的方式2021-06-06
MyBatis3源碼解析之如何獲取數(shù)據(jù)源詳解
用myBatis3與spring整合的時(shí)候,我們可以通過多種方式獲取數(shù)據(jù)源,下面這篇文章主要給大家介紹了關(guān)于MyBatis3源碼解析之如何獲取數(shù)據(jù)源的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06
java實(shí)現(xiàn)百度云文字識(shí)別接口代碼
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)百度云文字識(shí)別的接口代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11
mybatis plus動(dòng)態(tài)數(shù)據(jù)源切換及查詢過程淺析
這篇文章主要介紹了mybatis plus動(dòng)態(tài)數(shù)據(jù)源切換及查詢過程淺析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12

