成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Spring Security 密碼驗(yàn)證動(dòng)態(tài)加鹽的驗(yàn)證處理方法

瀏覽:40日期:2023-07-11 10:47:35

本文個(gè)人博客地址:https://www.leafage.top/posts/detail/21697I2R

最近幾天在改造項(xiàng)目,需要將gateway整合security在一起進(jìn)行認(rèn)證和鑒權(quán),之前gateway和auth是兩個(gè)服務(wù),auth是shiro寫的一個(gè),一個(gè)filter和一個(gè)配置,內(nèi)容很簡(jiǎn)單,生成token,驗(yàn)證token,沒有其他的安全檢查,然后讓對(duì)項(xiàng)目進(jìn)行重構(gòu)。

先是要整合gateway和shiro,然而因?yàn)間ateway是webflux,而shiro-spring是webmvc,所以沒搞成功,如果有做過并成功的,請(qǐng)告訴我如何進(jìn)行整合,非常感謝。

那整合security呢,因?yàn)閟pring cloud gateway基于webflux,所以網(wǎng)上很多教程是用不了的,webflux的配置會(huì)有一些變化,具體看如下代碼示例:

import io.leafage.gateway.api.HypervisorApi;import io.leafage.gateway.handler.ServerFailureHandler;import io.leafage.gateway.handler.ServerSuccessHandler;import io.leafage.gateway.service.JdbcReactiveUserDetailsService;import org.springframework.context.annotation.Bean;import org.springframework.http.HttpMethod;import org.springframework.http.HttpStatus;import org.springframework.security.config.annotation.web.reactive.EnableWebFluxSecurity;import org.springframework.security.config.web.server.ServerHttpSecurity;import org.springframework.security.core.userdetails.ReactiveUserDetailsService;import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;import org.springframework.security.crypto.password.PasswordEncoder;import org.springframework.security.web.server.SecurityWebFilterChain;import org.springframework.security.web.server.authentication.HttpStatusServerEntryPoint;import org.springframework.security.web.server.authentication.ServerAuthenticationFailureHandler;import org.springframework.security.web.server.authentication.ServerAuthenticationSuccessHandler;import org.springframework.security.web.server.authentication.logout.HttpStatusReturningServerLogoutSuccessHandler;import org.springframework.security.web.server.csrf.CookieServerCsrfTokenRepository;/** * spring security config . * * @author liwenqiang 2019/7/12 17:51 */@EnableWebFluxSecuritypublic class ServerSecurityConfiguration { // 用于獲取遠(yuǎn)程數(shù)據(jù) private final HypervisorApi hypervisorApi; public ServerSecurityConfiguration(HypervisorApi hypervisorApi) {this.hypervisorApi = hypervisorApi; } /** * 密碼配置,使用BCryptPasswordEncoder * * @return BCryptPasswordEncoder 加密方式 */ @Bean protected PasswordEncoder passwordEncoder() {return new BCryptPasswordEncoder(); } /** * 用戶數(shù)據(jù)加載 * * @return JdbcReactiveUserDetailsService 接口 */ @Bean public ReactiveUserDetailsService userDetailsService() {// 自定義的ReactiveUserDetails 實(shí)現(xiàn)return new JdbcReactiveUserDetailsService(hypervisorApi); } /** * 安全配置 */ @Bean SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) {http.formLogin(f -> f.authenticationSuccessHandler(authenticationSuccessHandler()).authenticationFailureHandler(authenticationFailureHandler())).logout(l -> l.logoutSuccessHandler(new HttpStatusReturningServerLogoutSuccessHandler())).csrf(c -> c.csrfTokenRepository(CookieServerCsrfTokenRepository.withHttpOnlyFalse())).authorizeExchange(a -> a.pathMatchers(HttpMethod.OPTIONS).permitAll().anyExchange().authenticated()).exceptionHandling(e -> e.authenticationEntryPoint(new HttpStatusServerEntryPoint(HttpStatus.UNAUTHORIZED)));return http.build(); } /** * 登陸成功后執(zhí)行的處理器 */ private ServerAuthenticationSuccessHandler authenticationSuccessHandler() {return new ServerSuccessHandler(); } /** * 登陸失敗后執(zhí)行的處理器 */ private ServerAuthenticationFailureHandler authenticationFailureHandler() {return new ServerFailureHandler(); }}

上面的示例代碼,是我開源項(xiàng)目中的一段,一般的配置就如上面寫的,就可以使用了,但是由于我們之前的項(xiàng)目中的是shiro,然后有一個(gè)自定義的加密解密的邏輯。

首先說明一下情況,之前那一套加密(前端MD5,不加鹽,然后數(shù)據(jù)庫(kù)存儲(chǔ)的是加鹽后的數(shù)據(jù)和對(duì)應(yīng)的鹽(每個(gè)賬號(hào)一個(gè)),要登錄比較之前對(duì)密碼要獲取動(dòng)態(tài)的鹽,然后加鹽進(jìn)行MD5,再進(jìn)行對(duì)比,但是在配置的時(shí)候是沒法獲取某一用戶的鹽值)

所以上面的一版配置是沒法通過驗(yàn)證的,必須在驗(yàn)證之前,給請(qǐng)求的密碼混合該賬號(hào)對(duì)應(yīng)的鹽進(jìn)行二次加密后在對(duì)比,但是這里就有問題了:

security 框架提供的幾個(gè)加密解密工具沒有MD5的方式; security 配置加密解密方式的時(shí)候,無法填入動(dòng)態(tài)的賬號(hào)的加密鹽;

對(duì)于第一個(gè)問題還好處理,解決方式是:自定義加密解密方式,然后注入到配置類中,示例如下:

import cn.hutool.crypto.SecureUtil;import com.ichinae.imis.gateway.utils.SaltUtil;import org.springframework.security.crypto.codec.Utf8;import org.springframework.security.crypto.password.PasswordEncoder;import java.security.MessageDigest;/** * 自定義加密解密 */public class MD5PasswordEncoder implements PasswordEncoder { @Override public String encode(CharSequence charSequence) {String salt = SaltUtil.generateSalt();return SecureUtil.md5(SecureUtil.md5(charSequence.toString()) + salt); } @Override public boolean matches(CharSequence charSequence, String encodedPassword) {byte[] expectedBytes = bytesUtf8(charSequence.toString());byte[] actualBytes = bytesUtf8(charSequence.toString());return MessageDigest.isEqual(expectedBytes, actualBytes); } private static byte[] bytesUtf8(String s) {// need to check if Utf8.encode() runs in constant time (probably not).// This may leak length of string.return (s != null) ? Utf8.encode(s) : null; }}

第二個(gè)問題的解決辦法,找了很多資料,也沒有找到,后來查看security的源碼發(fā)現(xiàn),可以在UserDetailsService接口的findByUsername()方法中,在返回UserDetails實(shí)現(xiàn)的時(shí)候,使用默認(rèn)實(shí)現(xiàn)User的UserBuilder內(nèi)部類來解決這個(gè)問題,因?yàn)閁serBuilder類中有一個(gè)屬性,passwordEncoder屬性,它是Fucntion<String, String>類型的,默認(rèn)實(shí)現(xiàn)是 password -> password,即對(duì)密碼不做任何處理,先看下它的源碼:

Spring Security 密碼驗(yàn)證動(dòng)態(tài)加鹽的驗(yàn)證處理方法

再看下解決問題之前的findByUsername()方法:

@Servicepublic class UserDetailsServiceImpl implements ReactiveUserDetailsService { @Resource private RemoteService remoteService; @Override public Mono<UserDetails> findByUsername(String username) {return remoteService.getUser(username).map(userBO -> User.builder().username(username).password(userBO.getPassword()).authorities(grantedAuthorities(userBO.getAuthorities())).build()); } private Set<GrantedAuthority> grantedAuthorities(Set<String> authorities) {return authorities.stream().map(SimpleGrantedAuthority::new).collect(Collectors.toSet()); }}

那找到了問題的解決方法,就來改代碼了,如下所示:

新增一個(gè)代碼處理方法

private Function<String, String> passwordEncoder(String salt) { return rawPassword -> SecureUtil.md5(rawPassword + salt);}

然后添加builder鏈

@Servicepublic class UserDetailsServiceImpl implements ReactiveUserDetailsService { @Resource private RemoteService remoteService; @Override public Mono<UserDetails> findByUsername(String username) {return remoteService.getUser(username).map(userBO -> User.builder().passwordEncoder(passwordEncoder(userBO.getSalt())) //在這里設(shè)置動(dòng)態(tài)的鹽.username(username).password(userBO.getPassword()).authorities(grantedAuthorities(userBO.getAuthorities())).build()); } private Set<GrantedAuthority> grantedAuthorities(Set<String> authorities) {return authorities.stream().map(SimpleGrantedAuthority::new).collect(Collectors.toSet()); } private Function<String, String> passwordEncoder(String salt) {return rawPassword -> SecureUtil.md5(rawPassword + salt); }}

然后跑一下代碼,請(qǐng)求登錄接口,就登陸成功了。

Spring Security 密碼驗(yàn)證動(dòng)態(tài)加鹽的驗(yàn)證處理方法

以上就是Spring Security 密碼驗(yàn)證動(dòng)態(tài)加鹽的驗(yàn)證處理的詳細(xì)內(nèi)容,更多關(guān)于Spring Security密碼驗(yàn)證的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
天天色天天爱天天射综合| 日韩不卡手机在线v区| 中文字幕综合网| 国产成人午夜精品影院观看视频 | 亚洲一区二区三区自拍| 欧美三级午夜理伦三级中文幕| 亚洲精品在线三区| 成人黄动漫网站免费app| 欧美一区欧美二区| 国产在线日韩欧美| 欧美三级韩国三级日本三斤| 欧美一级免费大片| 激情综合色播五月| 欧美三级电影网| 日韩国产精品大片| 久久国产直播| 亚洲bdsm女犯bdsm网站| 国产精品亚洲欧美| 一区二区激情视频| 国产精品日韩精品欧美精品| 亚洲精品国产a| 99国产精品99久久久久久粉嫩| 亚洲视频香蕉人妖| 在线看片欧美| 亚洲人吸女人奶水| 亚洲伦伦在线| 亚洲亚洲精品在线观看| 亚洲一区在线免费| 午夜久久久久久电影| 亚洲欧美不卡| 天天综合日日夜夜精品| 色综合久久99| 日本不卡一区二区| 欧美日韩免费观看一区三区| 国产一区二区三区在线观看免费视频| 337p亚洲精品色噜噜| 国产成人午夜精品5599 | 91丨porny丨国产入口| 国产日韩欧美精品一区| 香蕉av福利精品导航| 老鸭窝亚洲一区二区三区| 午夜在线电影亚洲一区| 色哟哟一区二区在线观看 | 美女一区二区三区在线观看| 欧美高清性hdvideosex| 成人激情午夜影院| 国产视频不卡一区| 亚洲天堂久久| 亚洲一区免费视频| 91久久线看在观草草青青| 国产在线一区二区综合免费视频| 精品剧情v国产在线观看在线| 91网站在线观看视频| 国产精品护士白丝一区av| 一区二区高清| 午夜国产精品一区| 欧美日韩视频在线观看一区二区三区| 成人av在线播放网站| 1024亚洲合集| 色婷婷国产精品| 国产传媒日韩欧美成人| 国产亚洲欧美色| 国产一区二区黄色| 精品一区二区三区av| 欧美日韩国产大片| 成人精品电影在线观看| 国产精品拍天天在线| 性感少妇一区| 久久av老司机精品网站导航| 8v天堂国产在线一区二区| jlzzjlzz欧美大全| 国产精品国产成人国产三级 | 久久久久亚洲蜜桃| 野花国产精品入口| 麻豆精品国产传媒mv男同| 精品国产亚洲一区二区三区在线观看| 欧美三级黄美女| 亚洲国产日韩在线一区模特 | 欧洲精品在线观看| 成人91在线观看| 亚洲三级免费观看| 欧美日韩大陆在线| 国产精品v欧美精品v日韩精品 | 欧美年轻男男videosbes| 欧美精品成人| 午夜精品久久久久久久蜜桃app| 欧美一区午夜精品| 国内精品视频在线播放| 日本伊人精品一区二区三区观看方式| 日韩精品在线一区二区| 99av国产精品欲麻豆| 狠狠色综合日日| 欧美精品一区二区三区在线播放| 国产偷久久久精品专区| 高清久久久久久| 亚洲美女视频一区| 欧美精品国产精品| 在线成人h网| 黑人巨大精品欧美一区| 一区二区中文视频| 欧美一区二区在线播放| 韩国av一区| 老司机精品视频在线| 国产精品国产成人国产三级| 欧美日韩在线三区| 亚洲视频一区| 国产精品一区二区久久不卡 | 国产精品免费一区二区三区在线观看 | 日本高清视频一区二区| 欧美凹凸一区二区三区视频| 日韩精品一二区| 国产蜜臀av在线一区二区三区 | 国产欧美日韩精品在线| 欧美日韩一区高清| 国内精品视频在线播放| 精品一区二区三区在线观看| 国产精品福利一区| 欧美日韩高清在线| 亚洲大黄网站| 国v精品久久久网| 亚洲.国产.中文慕字在线| 国产三级一区二区三区| 日本精品视频一区二区三区| 国产精品sss| 国产高清不卡二三区| 亚洲一区二区3| 精品91自产拍在线观看一区| 91福利视频久久久久| 亚洲免费激情| k8久久久一区二区三区| 青青草国产成人av片免费| 国产精品超碰97尤物18| 91精品国产综合久久久久久漫画| 欧美综合二区| 精品999日本| 97精品电影院| 国产精品一区二区免费不卡| 亚洲mv大片欧洲mv大片精品| 国产精品伦理在线| 欧美一区在线视频| 久久久久欧美精品| 亚洲国产高清一区二区三区| 国产伦精品一区二区三区视频青涩| 日韩理论电影院| 久久伊99综合婷婷久久伊| 欧美三级在线播放| 一本色道久久综合亚洲精品按摩| 韩国精品一区二区三区| eeuss鲁片一区二区三区在线观看| 日本午夜精品一区二区三区电影| 国产精品久久毛片av大全日韩| 日韩欧美一区在线| 欧美日韩三级在线| 在线观看成人免费视频| 蜜桃av久久久亚洲精品| 亚洲区一区二区三区| 欧美日韩一区二| 99久久精品国产观看| 国产成人亚洲精品狼色在线| 日本欧美加勒比视频| 有码一区二区三区| 中文字幕一区二区三中文字幕| 国产性天天综合网| 久久久.com| 精品国产凹凸成av人网站| 欧美一区日韩一区| 欧美精品乱码久久久久久按摩| 在线亚洲精品福利网址导航| 色哟哟日韩精品| 久久一区二区三区四区五区| 亚洲麻豆视频| 亚洲人人精品| 亚洲黄色av| 尤物在线精品| 亚洲区国产区| 国产亚洲欧美一区二区| 亚洲高清123| 亚洲毛片播放| 在线一区亚洲| 国产精品毛片| 欧美亚洲三区| 亚洲综合激情| 噜噜噜噜噜久久久久久91| 亚洲一区激情| 老司机久久99久久精品播放免费| 久久综合福利| 欧美主播一区二区三区美女| 91久久久免费一区二区| 在线观看日韩电影| 欧美少妇性性性| 91精品国产免费| 日韩一区二区电影| 欧美大片一区二区三区| 日韩欧美电影一区| 精品国产第一区二区三区观看体验| 久久综合丝袜日本网| 久久网站最新地址| 中文一区二区在线观看| 亚洲视频一区在线| 亚洲国产另类精品专区|