SpringSecurity 手機號登錄功能實現(xiàn)
一、工作流程
1.向手機發(fā)送驗證碼,第三方短信發(fā)送平臺,如阿里云短信。
2.手機獲取驗證碼后,在表單中輸入驗證碼。
3.使用自定義過濾器?SmsCodeValidateFilter?。
4.短信校驗通過后,使用自定義手機認證過濾器?SmsCodeAuthenticationFilter校驗手機號碼是否存在?。
5.自定義?SmsCodeAuthenticationToken?提供給?SmsCodeAuthenticationFilter?。
6.自定義?SmsCodeAuthenticationProvider?提供給?AuthenticationManager?。
7.創(chuàng)建針對手機號查詢用戶信息的?SmsCodeUserDetailsService?,提交給?。SmsCodeAuthenticationProvider?。
8.自定義?SmsCodeSecurityConfig?配置類將上面組件連接起來。
9.將?SmsCodeSecurityConfig?添加到?LearnSrpingSecurity?安全配置的過濾器鏈上。
二、實現(xiàn)
2.1、驗證碼生成、發(fā)送
/** * 創(chuàng)建驗證碼生成器 */ @Component public class SmsCodeGenerator { public String generate() { return RandomStringUtils.randomNumeric(4); } } /** * 驗證碼發(fā)送器 */ @Component public class SmsCodeSender { public void send(String mobile, String code) { System.out.println("向手機" + mobile + "發(fā)送短信驗證碼" + code); } } /** * 發(fā)送短信接口 */ @RestController public class ValidateCodeController { @Autowired private SmsCodeGenerator smsCodeGenerator; @Resource private SmsCodeSender smsCodeSender; @Resource private RedisTemplate redisTemplate; @GetMapping("/code/sms") public String createSmsCode(@RequestParam String mobile) throws IOException { //獲取驗證碼 String smsCode = smsCodeGenerator.generate(); //把驗證碼設(shè)置到redis redisTemplate.boundValueOps(SecurityConstants.getValidCodeKey(mobile)).set(smsCode, 300, TimeUnit.SECONDS); smsCodeSender.send("18360903475", "登錄驗證碼為:" + smsCode + ",五分鐘過期"); return "驗證碼是 : " + smsCode; } }
2.2、手機號碼認證 Token
import org.springframework.security.authentication.AbstractAuthenticationToken; import org.springframework.security.core.GrantedAuthority; import org.springframework.security.core.SpringSecurityCoreVersion; import java.util.Collection; /** * 手機號碼認證 Token */ public class PhoneNumAuthenticationToken extends AbstractAuthenticationToken { private static final long serialVersionUID = SpringSecurityCoreVersion.SERIAL_VERSION_UID; /** * principal的作用有兩個, 在未登錄之前是用戶名,那么在登錄之后是用戶的信息。 */ private final Object principal; /** * 構(gòu)造 * @param principal 手機號碼 */ public PhoneNumAuthenticationToken(Object principal) { super(null); this.principal = principal; // 用于指示AbstractSecurityInterceptor是否應(yīng)向AuthenticationManager提供身份驗證令牌 setAuthenticated(false); } /** * 構(gòu)造 * @param principal 用戶信息 * @param authorities 用戶權(quán)限列表 */ public PhoneNumAuthenticationToken(Object principal,Collection<? extends GrantedAuthority> authorities) { super(authorities); this.principal = principal; // 用于指示AbstractSecurityInterceptor是否應(yīng)向AuthenticationManager提供身份驗證令牌 setAuthenticated(true); } /** * 正常這個是返回密碼,但手機登錄沒有密碼,不用管 */ @Override public Object getCredentials() { return null; } /** * 獲取手機號或用戶信息 */ @Override public Object getPrincipal() { return this.principal; } }
2.3、攔截請求、獲取手機號碼
/** * 手機號碼攔截器, 獲取手機號碼 */ public class PhoneNumAuthenticationFilter extends AbstractAuthenticationProcessingFilter { public PhoneNumAuthenticationFilter() { super(new AntPathRequestMatcher("/phoneLogin", "POST")); } @Override public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response) throws AuthenticationException { if (!Objects.equals(request.getMethod(),"POST")) { throw new AuthenticationServiceException("身份驗證方法需為:'POST'請求"); } // 獲取手機號 String phoneNum = Optional.ofNullable(request.getParameter(Constants.PHONE_NUM_PARAMETER)).map(String::trim).orElse(""); // new 手機號碼驗證Token PhoneNumAuthenticationToken authRequest = new PhoneNumAuthenticationToken(phoneNum); // 身份驗證詳細信息 authRequest.setDetails(super.authenticationDetailsSource.buildDetails(request)); return this.getAuthenticationManager().authenticate(authRequest); } }
2.4、短信驗證碼驗證過濾器
/** * 短信驗證碼驗證過濾器 */ @Component public class SmsCodeFilter extends OncePerRequestFilter { @Resource private StringRedisTemplate stringRedisTemplate; @Resource private CustomizeAuthencationFailureHandler customizeAuthencationFailureHandler; @Override protected void doFilterInternal(HttpServletRequest request,HttpServletResponse response,FilterChain filterChain) throws ServletException, IOException { /** * uri = /phoneLogin 即手機號碼登錄才攔截 */ if (Objects.equals(Constants.SMS_LOGIN_URI,request.getRequestURI())) { try{ // 驗證手機驗證碼 validateProcess(request); }catch (AuthenticationException ex) { customizeAuthencationFailureHandler.onAuthenticationFailure(request, response, ex); return; } } filterChain.doFilter(request, response); } /** * 驗證手機驗證碼 */ private void validateProcess(HttpServletRequest request){ // 獲取手機號 String msgCode = stringRedisTemplate.opsForValue().get(Constants.SMS_CODE_SESSION_KEY); String code = request.getParameter(Constants.MSG_CODE); if(Strings.isBlank(code)) { throw new InternalAuthenticationServiceException("短信驗證碼不能為空."); } if(null == msgCode) { throw new InternalAuthenticationServiceException("短信驗證碼已失效."); } if(!code.equals(msgCode)) { throw new InternalAuthenticationServiceException("短信驗證碼錯誤."); } } }
2.5、繼承 WebSecurityConfigurerAdapter 配置 HttpSecurity
@Configuration public class SecurityConfig extends WebSecurityConfigurerAdapter { /** * 數(shù)據(jù)源 */ @Resource private DataSource dataSource; /** * 用戶信息服務(wù) */ @Resource private UserAuthentication userAuthentication; /** * 成功處理 */ @Resource private CustomizeAuthencationSuccessHandler customizeAuthencationSuccessHandler; /** * 失敗處理 */ @Resource private CustomizeAuthencationFailureHandler customizeAuthencationFailureHandler; /** * 用戶登出處理 */ @Resource private UserLogoutSuccessHandler userLogoutSuccessHandler; /** * 多用戶登錄處理 */ @Resource private MutilpleSessionHandler mutilpleSessionHandler; /** * 密碼編碼器 */ @Bean public PasswordEncoder passwordEncoder() { return new BCryptPasswordEncoder(); } /** * 手機號碼登錄驗證處理 */ @Resource private DaoPhoneNumAuthenticationProvider daoPhoneNumAuthenticationProvider; /** * 信息驗證碼過濾器 */ @Resource private SmsCodeFilter smsCodeFilter; /** * 把AuthenticationManager公開 */ @Override @Bean public AuthenticationManager authenticationManagerBean() throws Exception { return super.authenticationManagerBean(); } /** * 配置自定義驗證查詢/加密工具 */ @Override protected void configure(AuthenticationManagerBuilder auth) throws Exception { auth.userDetailsService(userAuthentication).passwordEncoder(passwordEncoder()); } /** * 手機號碼登錄攔截器 */ @Bean public PhoneNumAuthenticationFilter phoneNumAuthenticationFilter() throws Exception { // 手機號碼攔截器, 獲取手機號碼 PhoneNumAuthenticationFilter phoneNumAuthenticationFilter = new PhoneNumAuthenticationFilter(); phoneNumAuthenticationFilter.setAuthenticationManager(authenticationManagerBean()); //使用手機號登錄失敗了如何處理 phoneNumAuthenticationFilter.setAuthenticationFailureHandler(customizeAuthencationFailureHandler); // 使用手機號登錄成功了如何處理 phoneNumAuthenticationFilter.setAuthenticationSuccessHandler(customizeAuthencationSuccessHandler); return phoneNumAuthenticationFilter; } @Override protected void configure(HttpSecurity http) throws Exception { http // 加入短信驗證碼過濾器 .addFilterBefore(smsCodeFilter, UsernamePasswordAuthenticationFilter.class) // 加入手機號碼登錄過濾器 .addFilterAfter(phoneNumAuthenticationFilter(), UsernamePasswordAuthenticationFilter.class) // 加入手機號碼登錄驗證提供者 .authenticationProvider(daoPhoneNumAuthenticationProvider) // 表單登錄 .formLogin() // 未登錄跳轉(zhuǎn)登錄頁面 .loginPage("/login.html") // 指定登錄路徑 .loginProcessingUrl("/login") // 用戶登錄成功的處理 .successHandler(customizeAuthencationSuccessHandler) // 用戶登錄失敗的處理 .failureHandler(customizeAuthencationFailureHandler) // 因為用戶傳入過來的token, 需要再次進行校驗 .userDetailsService(userAuthentication) .tokenValiditySeconds(3600) // .alwaysRemember(true) // 認證配置 .and() .authorizeRequests() //不攔截的Url .antMatchers("/login.html", "/image/code", "/smsCode", "/css/**", "/js/**", "/phoneLogin").permitAll() .anyRequest() //所有的請求 .authenticated() //認證之后就可以訪問 // 多端登錄限制,限制一個賬號同時只能一個人登錄 .and() .sessionManagement() .maximumSessions(1) .expiredSessionStrategy(mutilpleSessionHandler) .and() // 登出配置 .and() .logout() .logoutUrl("/logout") // 登出成功處理 .logoutSuccessHandler(userLogoutSuccessHandler) .and() .csrf().disable(); } }
到此這篇關(guān)于SpringSecurity 手機號登錄的文章就介紹到這了,更多相關(guān)SpringSecurity 手機號登錄內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringSecurity整合springBoot、redis實現(xiàn)登錄互踢功能
- SpringSecurity構(gòu)建基于JWT的登錄認證實現(xiàn)
- springboot+jwt+springSecurity微信小程序授權(quán)登錄問題
- 詳解SpringSecurity中的Authentication信息與登錄流程
- SpringBoot整合SpringSecurity和JWT的示例
- SpringSecurity 默認表單登錄頁展示流程源碼
- SpringSecurity動態(tài)加載用戶角色權(quán)限實現(xiàn)登錄及鑒權(quán)功能
- 解析SpringSecurity自定義登錄驗證成功與失敗的結(jié)果處理問題
相關(guān)文章
SpringCloud消息總線Bus配置中心實現(xiàn)過程解析
這篇文章主要介紹了SpringCloud消息總線Bus配置中心實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-03-03關(guān)于springboot2整合lettuce啟動卡住問題的解決方法
Lettuce和Jedis的都是連接Redis Server的客戶端程序,下面這篇文章主要給大家介紹了關(guān)于springboot2整合lettuce啟動卡住問題的解決方法,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下2021-12-12SpringBoot依賴和代碼分開打包的實現(xiàn)步驟
本文主要介紹了SpringBoot依賴和代碼分開打包的實現(xiàn)步驟,,這種方法將依賴和代碼分開打包,一般更新只有代碼修改,Pom文件是不會經(jīng)常改動的,感興趣的可以了解一下2023-10-10springboot中使用過濾器,jsoup過濾XSS腳本詳解
這篇文章主要介紹了springboot中使用過濾器,jsoup過濾XSS腳本詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12[Spring MVC]-詳解SpringMVC的各種參數(shù)綁定方式
本篇文章主要介紹了SpringMVC的各種參數(shù)綁定方式 ,具有一定的參考價值,有需要的可以了解一下。2016-12-12使用Spring?Boot+gRPC構(gòu)建微服務(wù)并部署的案例詳解
這篇文章主要介紹了使用Spring?Boot+gRPC構(gòu)建微服務(wù)并部署,Spring Cloud僅僅是一個開發(fā)框架,沒有實現(xiàn)微服務(wù)所必須的服務(wù)調(diào)度、資源分配等功能,這些需求要借助Kubernetes等平臺來完成,本文給大家介紹的非常詳細,需要的朋友參考下吧2022-06-06SQLSyntaxErrorException-ExecutorException報錯解決分析
這篇文章主要為大家介紹了SQLSyntaxErrorException-ExecutorException報錯解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08