一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - Spring Security中用JWT退出登錄時(shí)遇到的坑

Spring Security中用JWT退出登錄時(shí)遇到的坑

2022-02-21 13:08碼農(nóng)小胖哥 Java教程

使用了JWT后,每次請(qǐng)求都要攜帶 Bearer Token 并且被專門的過濾器攔截解析之后才能將用戶認(rèn)證信息保存到 SecurityContext 中去,接下來通過本文給大家介紹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)邏輯為:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// 當(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 過濾器排序圖:

Spring Security中用JWT退出登錄時(shí)遇到的坑

也就說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)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!

原文鏈接:https://blog.didispace.com/spring-security-jwt-logout-wrong-config

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 蜜桃影像传媒推广 | 99综合在线 | 222aaa精品影院| 国产在线视频自拍 | 国内自拍网红在综合图区 | crdy在线看亚洲 | 视频免费 | 护士柔佳 | 日韩精品一区二三区中文 | 含羞草传媒一天免费看下 | 91香蕉国产在线观看免费永久 | 国产美女极品免费视频 | 国产精品99在线观看 | 96日本xxxxxxxxx70 95在线观看精品视频 | 成年人视频在线播放 | 男人日女人的逼视频 | xxx中国bbbwww| 国产剧情一区 | 四虎成人免费观看在线网址 | 全色黄大色黄大片爽一次 | 水多多www视频在线观看高清 | 色综合天天娱乐综合网 | 美女大鸡鸡 | 精品乱lun小说 | 91精品国产91热久久久久福利 | 奇米影视久久 | 午夜伦理yy44008影院 | 国产麻豆剧果冻传媒观看免费视频 | 青青草国产免费国产是公开 | 无耻之徒第十一季在线观看 | 欧美肥乳| 91碰| 刺激一区仑乱 | 贰佰麻豆剧果冻传媒一二三区 | 好男人资源在线观看免费的 | 成人黄页网站 | chinesemature丰满成熟 | 亚洲大片在线观看 | 午夜影院和视费x看 | 极品久久 | 免费一区|