日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Spring Security 實現“記住我”功能及原理解析

瀏覽:123日期:2023-09-05 08:05:36

這章繼續擴展功能,來一個“記住我”的功能實現,就是說用戶在登錄一次以后,系統會記住這個用戶一段時間,這段時間內用戶不需要重新登錄就可以使用系統。

記住我功能基本原理

原理說明

用戶登錄發送認證請求的時候會被UsernamePasswordAuthenticationFilter認證攔截,認證成功以后會調用一個RememberMeService服務,服務里面有一個TokenRepository,這個服務會生成一個Token,然后將Token寫入到瀏覽器的Cookie同時會使用TokenRepository把生成的Token寫到數據庫里面,因為這個動作是在認證成功以后做的,所以在Token寫入數據庫的時候會把用戶名同時寫入數據庫。 假如瀏覽器關了重新訪問系統,用戶不需要再次登錄就可以訪問,這個時候請求在過濾器鏈上會經過RememberMeAuthenticationFilter,這個過濾器的作用是讀取Cookie中的Token交給RemeberMeService,RemeberMeService會用TokenRepository到數據庫里去查這個Token在數據庫里有沒有記錄,如果有記錄就會把用戶名取出來,取出來以后會進行各種校驗然后生成新Token再調用之前的UserDetailService,去獲取用戶的信息,然后把用戶信息放到SecurityContext里面,到這里就把用戶給登錄上了。

圖解說明

Spring Security 實現“記住我”功能及原理解析

RememberMeAuthenticationFilter位于過濾器鏈的哪一環?

圖解

Spring Security 實現“記住我”功能及原理解析

首先其他認證過濾器會先進行認證,當其他過濾器都無法認證時,RememberMeAuthenticationFilter會嘗試去做認證。

記住我功能具體實現

前端頁面

登錄的時候加上一行記住我的勾選按鈕,這里要注意,name一定要是remember-me,下面源碼部分會提到。

<tr><td colspan=’2’><input name='remember-me' type='checkbox' value='true' />記住我</td></tr>

后臺

首先配置TokenRepositoryBean

/** * 記住我功能的Token存取器配置 * * @return */@Beanpublic PersistentTokenRepository persistentTokenRepository() {JdbcTokenRepositoryImpl tokenRepository = new JdbcTokenRepositoryImpl();tokenRepository.setDataSource(dataSource);// 啟動的時候自動創建表,建表語句 JdbcTokenRepositoryImpl 已經都寫好了tokenRepository.setCreateTableOnStartup(true);return tokenRepository;}

然后需要在 configure 配置方法那邊進行記住我功能所有組件的配置

protected void configure(HttpSecurity http) throws Exception {ValidateCodeFilter validateCodeFilter = new ValidateCodeFilter();http.addFilterBefore(validateCodeFilter, UsernamePasswordAuthenticationFilter.class).formLogin().loginPage('/authentication/require').loginProcessingUrl('/authentication/form').successHandler(meicloudAuthenticationSuccessHandler).failureHandler(meicloudAuthenticationFailureHandler)// 配置記住我功能.and().rememberMe()// 配置TokenRepository.tokenRepository(persistentTokenRepository())// 配置Token過期時間.tokenValiditySeconds(3600)// 最終拿到用戶名之后,使用UserDetailsService去做登錄.userDetailsService(userDetailsService).and().authorizeRequests().antMatchers('/authentication/require', securityProperties.getBrowser().getSignInPage(), '/code/image').permitAll().anyRequest().authenticated().and().csrf().disable();}

記住我功能Spring Security源碼解析

登錄之前“記住我”源碼流程

在認證成功之后,會調用successfulAuthentication方法(這些第五章源碼部分已經學習過),在將認證信息保存到Context后,RememberMeServices就會調用它的loginSuccess方法。

protected void successfulAuthentication(HttpServletRequest request, HttpServletResponse response, FilterChain chain, Authentication authResult) throws IOException, ServletException { if (this.logger.isDebugEnabled()) { this.logger.debug('Authentication success. Updating SecurityContextHolder to contain: ' + authResult); } SecurityContextHolder.getContext().setAuthentication(authResult); this.rememberMeServices.loginSuccess(request, response, authResult); if (this.eventPublisher != null) { this.eventPublisher.publishEvent(new InteractiveAuthenticationSuccessEvent(authResult, this.getClass())); } this.successHandler.onAuthenticationSuccess(request, response, authResult); }

loginSuccess方法里面會先檢查請求中是否有name為remember-me的參數,有才進行下一步。

public final void loginSuccess(HttpServletRequest request, HttpServletResponse response, Authentication successfulAuthentication) { // this.parameter = 'remember-me' if (!this.rememberMeRequested(request, this.parameter)) { this.logger.debug('Remember-me login not requested.'); } else { this.onLoginSuccess(request, response, successfulAuthentication); } }

再進入onLoginSuccess方法,里面主要就是進行寫庫和寫Cookie的操作。

protected void onLoginSuccess(HttpServletRequest request, HttpServletResponse response, Authentication successfulAuthentication) { String username = successfulAuthentication.getName(); this.logger.debug('Creating new persistent login for user ' + username); // 生成Token PersistentRememberMeToken persistentToken = new PersistentRememberMeToken(username, this.generateSeriesData(), this.generateTokenData(), new Date()); try { // 將Token和userName插入數據庫 this.tokenRepository.createNewToken(persistentToken); // 將Token寫到Cookie中 this.addCookie(persistentToken, request, response); } catch (Exception var7) { this.logger.error('Failed to save persistent token ', var7); } }

登錄之后“記住我”源碼流程

首先會進入RememberMeAuthenticationFilter,會先判斷前面的過濾器是否進行過認證(Context中是否有認證信息),未進行過認證的話會調用RememberMeServices的autoLogin方法。

public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest)req; HttpServletResponse response = (HttpServletResponse)res; if (SecurityContextHolder.getContext().getAuthentication() == null) { Authentication rememberMeAuth = this.rememberMeServices.autoLogin(request, response); if (rememberMeAuth != null) { try { rememberMeAuth = this.authenticationManager.authenticate(rememberMeAuth); SecurityContextHolder.getContext().setAuthentication(rememberMeAuth); this.onSuccessfulAuthentication(request, response, rememberMeAuth); if (this.logger.isDebugEnabled()) { this.logger.debug('SecurityContextHolder populated with remember-me token: ’' + SecurityContextHolder.getContext().getAuthentication() + '’'); } if (this.eventPublisher != null) { this.eventPublisher.publishEvent(new InteractiveAuthenticationSuccessEvent(SecurityContextHolder.getContext().getAuthentication(), this.getClass())); } if (this.successHandler != null) { this.successHandler.onAuthenticationSuccess(request, response, rememberMeAuth); return; } } catch (AuthenticationException var8) { if (this.logger.isDebugEnabled()) { this.logger.debug('SecurityContextHolder not populated with remember-me token, as AuthenticationManager rejected Authentication returned by RememberMeServices: ’' + rememberMeAuth + '’; invalidating remember-me token', var8); } this.rememberMeServices.loginFail(request, response); this.onUnsuccessfulAuthentication(request, response, var8); } } chain.doFilter(request, response); } else { if (this.logger.isDebugEnabled()) { this.logger.debug('SecurityContextHolder not populated with remember-me token, as it already contained: ’' + SecurityContextHolder.getContext().getAuthentication() + '’'); } chain.doFilter(request, response); } }

autoLogin方法里面,主要調用this.processAutoLoginCookie(cookieTokens, request, response)這個方法獲取數據庫中的用戶信息,其步驟是:

解析前端傳來的Cookie,里面包含了Token和seriesId,它會使用seriesId查找數據庫的Token 檢查Cookie中的Token和數據庫查出來的Token是否一樣 一樣的話再檢查數據庫中的Token是否已過期 如果以上都符合的話,會使用舊的用戶名和series重新new一個Token,這時過期時間也重新刷新 然后將新的Token保存回數據庫,同時添加回Cookie中 最后再調用UserDetailsService的loadUserByUsername方法返回UserDetails

protected UserDetails processAutoLoginCookie(String[] cookieTokens, HttpServletRequest request, HttpServletResponse response) { if (cookieTokens.length != 2) { throw new InvalidCookieException('Cookie token did not contain 2 tokens, but contained ’' + Arrays.asList(cookieTokens) + '’'); } else { String presentedSeries = cookieTokens[0]; String presentedToken = cookieTokens[1]; PersistentRememberMeToken token = this.tokenRepository.getTokenForSeries(presentedSeries); if (token == null) { throw new RememberMeAuthenticationException('No persistent token found for series id: ' + presentedSeries); } else if (!presentedToken.equals(token.getTokenValue())) { this.tokenRepository.removeUserTokens(token.getUsername()); throw new CookieTheftException(this.messages.getMessage('PersistentTokenBasedRememberMeServices.cookieStolen', 'Invalid remember-me token (Series/token) mismatch. Implies previous cookie theft attack.')); } else if (token.getDate().getTime() + (long)this.getTokenValiditySeconds() * 1000L < System.currentTimeMillis()) { throw new RememberMeAuthenticationException('Remember-me login has expired'); } else { if (this.logger.isDebugEnabled()) { this.logger.debug('Refreshing persistent login token for user ’' + token.getUsername() + '’, series ’' + token.getSeries() + '’'); } PersistentRememberMeToken newToken = new PersistentRememberMeToken(token.getUsername(), token.getSeries(), this.generateTokenData(), new Date()); try { this.tokenRepository.updateToken(newToken.getSeries(), newToken.getTokenValue(), newToken.getDate()); this.addCookie(newToken, request, response); } catch (Exception var9) { this.logger.error('Failed to update token: ', var9); throw new RememberMeAuthenticationException('Autologin failed due to data access problem'); } return this.getUserDetailsService().loadUserByUsername(token.getUsername()); } } }

回到RememberMeAuthenticationFilter,在調用了autoLogin方法之后得到了rememberMeAuth,然后再對其進行一個認證,認證成功之后保存到SecurityContext中,至此整個RememberMe自動登錄流程源碼結束。

相關閱讀:

Spring Security實現圖形驗證碼登錄

Spring Security實現短信驗證碼登錄

總結

到此這篇關于Spring Security 實現“記住我”功能及原理解析的文章就介紹到這了,更多相關spring security記住我內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲激情二区| 亚洲中午字幕| 麻豆精品在线| 麻豆国产欧美一区二区三区 | 国产福利一区二区精品秒拍| 麻豆一区二区99久久久久| 日韩精品电影| 久久亚洲欧洲| 精品国产中文字幕第一页 | 欧美亚洲国产精品久久| 蜜臀av性久久久久蜜臀aⅴ四虎| 国产日韩一区二区三区在线播放| 国产精品毛片一区二区在线看| 亚洲大片在线| 国产视频一区二区在线播放| 精品欧美一区二区三区在线观看| 日韩一区中文| 激情久久久久久| 国产视频一区三区| 中文字幕一区二区三区日韩精品 | 国产亚洲在线| 色综合视频一区二区三区日韩 | 久久精品国产www456c0m| 视频在线观看一区二区三区| 久久中文字幕导航| 91九色精品| 久久精品国产久精国产| 日韩和欧美的一区| 国产精品麻豆成人av电影艾秋| 九九99久久精品在免费线bt| 久久精品1区| 久久精品伊人| 99视频精品全国免费| 欧美日韩a区| 不卡中文一二三区| 国产精品日本一区二区三区在线| 精品一区亚洲| 久久精品资源| 99国产精品久久久久久久| 精品久久一区| 精品国产美女a久久9999| 色老板在线视频一区二区| 亚洲国产成人精品女人| 欧美日本一区| 亚洲v在线看| 日韩和欧美的一区| 日韩在线高清| 蜜臀av一区二区在线免费观看| 精品中文在线| 国产精品日韩| 久久精品国产999大香线蕉| re久久精品视频| 国产伦精品一区二区三区在线播放| 国产精品成人a在线观看| 国产伊人精品| 国产精品美女久久久久久不卡| 一区二区三区四区在线看| 99久久婷婷| 在线精品一区| 亚洲欧美日韩国产一区二区| 日韩另类视频| 国产精品99视频| 国产精品66| 91免费精品国偷自产在线在线| 久色成人在线| 亚洲激情偷拍| 国产99久久| 亚洲精品88| 国产日韩欧美中文在线| 在线精品视频一区| 99国产精品久久久久久久| 久久精品在线| 日本免费久久| 国产精品国产一区| 国产精品一区二区免费福利视频| 日韩有吗在线观看| 亚洲aa在线| 免费日韩av片| 国产精品婷婷| av成人国产| 亚洲欧洲一区| 99热免费精品| 午夜一级久久| av最新在线| 国产综合色产| 欧美国产偷国产精品三区| 91嫩草精品| 水野朝阳av一区二区三区| 999国产精品视频| 福利一区和二区| 国产女人18毛片水真多18精品| 老司机久久99久久精品播放免费| 99精品在线免费在线观看| 精品久久91| 国产精品极品| 国产精品一区二区三区美女 | 久久成人精品| 久久久9色精品国产一区二区三区| 麻豆一区二区三| 欧美亚洲三级| 日本在线视频一区二区| 男人的天堂久久精品| 视频一区视频二区中文| 三级亚洲高清视频| 蜜桃久久精品一区二区| 亚洲va久久| 欧美一区精品| 久久这里只有精品一区二区| 97精品在线| 欧美va天堂在线| 亚洲一区欧美激情| 亚洲日本免费电影| 亚洲日产国产精品| 91成人在线精品视频| 久久gogo国模啪啪裸体| 国产成人久久精品麻豆二区 | 精品视频在线观看网站| av中文字幕在线观看第一页| 91看片一区| 欧美日韩国产综合网| 视频一区中文字幕精品| 欧美精品97| 麻豆网站免费在线观看| 免费视频亚洲| 日韩av午夜在线观看| 尤物在线精品| 国产精选在线| 亚洲精品看片| 日产欧产美韩系列久久99| 亚洲伊人影院| 亚洲精品三级| 久久精品999| 欧美91在线|欧美| 欧美www视频在线观看| 新版的欧美在线视频| 私拍精品福利视频在线一区| 亚洲午夜精品久久久久久app| 性欧美69xoxoxoxo| 99热精品在线| 亚洲综合欧美| 偷拍亚洲精品| 国产亚洲一卡2卡3卡4卡新区| 国产精品视频一区二区三区四蜜臂 | 国产精品免费大片| 精品入口麻豆88视频| 国产精品99一区二区三| 久久精品亚洲欧美日韩精品中文字幕| 91成人超碰| 日韩中文av| 国产精品免费大片| 日韩国产一区二区| 久久国产成人| 精品在线播放| 日韩精品a在线观看91| 久久精品系列| 蜜桃成人av| 婷婷视频一区二区三区| 久久这里只有| 婷婷精品视频| 亚洲精选91| 国产精品一区二区免费福利视频 | 蜜臀av一区二区在线免费观看| 欧美日韩一区自拍| 中文字幕在线高清| 视频在线在亚洲| 你懂的国产精品永久在线| 久久婷婷av| 日本综合视频| 丁香六月综合| 视频一区在线播放| 美女精品久久| 激情婷婷综合| 亚洲一二三区视频| 粉嫩av一区二区三区四区五区 | 国产一区二区中文| 午夜精品影视国产一区在线麻豆| 久久av影院| 久久亚洲国产| 日韩av不卡在线观看| 日韩欧美二区| 蜜桃视频在线观看一区二区| 美女视频免费精品| 欧美日韩亚洲一区在线观看| 国产日韩一区| 老牛影视一区二区三区| 国产亚洲一区二区手机在线观看| 久久狠狠亚洲综合| 蜜臀久久久99精品久久久久久| 欧美二三四区| 久久伊人国产| 日本三级亚洲精品| 亚洲主播在线| 久久天堂成人| 福利视频一区| 国产高清亚洲| 日本国产欧美| 亚洲日产国产精品| 国产一级一区二区| 九色porny丨国产首页在线| 久久精品毛片|