广告
返回顶部
首页 > 资讯 > 精选 >Spring Security如何实现自动登陆功能
  • 581
分享到

Spring Security如何实现自动登陆功能

2023-06-25 12:06:13 581人浏览 八月长安
摘要

这篇文章主要介绍spring Security如何实现自动登陆功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!当我们在登录像QQ邮箱这种大多数的网站,往往在登录按键上会有下次自动登录这个选项,勾选后登录成功,在一段

这篇文章主要介绍spring Security如何实现自动登陆功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

当我们在登录像QQ邮箱这种大多数的网站,往往在登录按键上会有下次自动登录这个选项,勾选后登录成功,在一段时间内,即便退出浏览器或者服务器重启,再次访问不需要用户输入账号密码进行登录,这也解决了用户每次输入账号密码的麻烦。

Spring Security如何实现自动登陆功能

接下来实现自动登陆。

applicatio.properties配置用户名密码

spring.security.user.name=javaspring.security.user.passWord=java

controller层实现

@RestControllerpublic class HelloController {    @GetMapping("/hello")    public String hello() {        return "hello";    }}

配置类实现

@Configurationpublic class SecurityConfig extends WEBSecurityConfigurerAdapter {    @Override    protected void configure(httpsecurity Http) throws Exception {    http.fORMLogin()            .and()            .authorizeRequests()            .anyRequest()            .authenticated()            .and()            .rememberMe()            .and()            .csrf().disable();}

访问http://localhost:8080/hello,此时系统会重定向到登录页面。

Spring Security如何实现自动登陆功能

二话不说,输入账号密码,开搞!

此时看到了登录数据remember-me的值为on,当自定义登陆框的时候应该知道如何定义key了吧。

Spring Security如何实现自动登陆功能

在hello接口,可以很清楚的看到cookie里保存了一个remember-me的令牌,这个就是自动登录的关键所在。

Spring Security如何实现自动登陆功能

至于令牌是怎么生成的,先看一段源码。核心处理类TokenBasedRememberMeServices->onLoginSuccess

public void onLoginSuccess(HttpServletRequest request, HttpServletResponse response, Authentication successfulAuthentication) {    //拿到用户名和密码    String username = this.retrieveUserName(successfulAuthentication);    String password = this.retrievePassword(successfulAuthentication);    //用户名为空 打印日志    if (!StringUtils.hasLength(username)) {        this.logger.debug("Unable to retrieve username");    } else {        //密码为空 通过用户名再去查询        if (!StringUtils.hasLength(password)) {            UserDetails user = this.getUserDetailsService().loadUserByUsername(username);            password = user.getPassword();            //查到的密码还为空 打印日志 结束            if (!StringUtils.hasLength(password)) {                this.logger.debug("Unable to obtain password for user: " + username);                return;            }        }        //令牌有效期的生成 1209600是两周 也就是说令牌有效期14天        int tokenLifetime = this.calculateLoginLifetime(request, successfulAuthentication);        long expiryTime = System.currentTimeMillis();        expiryTime += 1000L * (long)(tokenLifetime < 0 ? 1209600 : tokenLifetime);        //生成签名 signature        String signatureValue = this.makeTokenSignature(expiryTime, username, password);        //设置cookie        this.setCookie(new String[]{username, Long.toString(expiryTime), signatureValue}, tokenLifetime, request, response);        if (this.logger.isDebugEnabled()) {            this.logger.debug("Added remember-me cookie for user '" + username + "', expiry: '" + new Date(expiryTime) + "'");        }    }} //使用MD5加密 通过用户名、令牌有效期、密码和key生成rememberMe的令牌 这里的key也就是加密的盐值protected String makeTokenSignature(long tokenExpiryTime, String username, String password) {    String data = username + ":" + tokenExpiryTime + ":" + password + ":" + this.geTKEy();    try {        MessageDigest digest = MessageDigest.getInstance("MD5");        return new String(Hex.encode(digest.digest(data.getBytes())));    } catch (NoSuchAlGorithmException var7) {        throw new IllegalStateException("No MD5 algorithm available!");    }}

看完了核心的源码,也就知道了令牌的生成规则:username + “:” + tokenExpiryTime + “:” + password + “:” + key(key 是一个散列盐值,可以用来防治令牌被修改,通过MD5散列函数生成。),然后通过Base64编码。

取出刚才的remember-me=amF2YToxNjM3MTI2MDk1OTMxOMQ5OGI3OTY5OTE4ZmQwMzE3ZWUyY2U4Y2MzMjQxZGQ0进行下验证。

Spring Security如何实现自动登陆功能

解码后是java:1637126095931:d98b7969918fd0317ee2ce8cc3241dd4,很明显javausername1637126095931是两周后的tokenExpiryTimed98b7969918fd0317ee2ce8cc3241dd4passwordkey值的MD5加密生成的。

需要注意的是key值是通过UUID随机生成的,当重启服务器时,UUID的变化会导致自动登录失败,所以为了避免之前生成的令牌失效,可以在配置中定义key值。

@Overrideprotected void configure(HttpSecurity http) throws Exception {    http.formLogin()            .and()            .authorizeRequests()            .anyRequest()            .authenticated()            .and()            .rememberMe()            .key("HelloWorld")            .and()            .csrf().disable();}

在Spring Security—登陆流程分析曾经说到 Spring Security中的认证授权都是通过过滤器来实现的。RememberMeAuthenticationFilter 是自动登录的核心过滤器。

public class RememberMeAuthenticationFilter extends GenericFilterBean implements ApplicationEventPublisherAware {    private void doFilter(HttpServletRequest request, HttpServletResponse response, FilterChain chain)      throws IOException, ServletException {        //获取当前用户实例 继续过滤校验   if (SecurityContextHolder.getContext().getAuthentication() != null) {      this.logger.debug(LogMessage            .of(() -> "SecurityContextHolder not populated with remember-me token, as it already contained: '"                  + SecurityContextHolder.getContext().getAuthentication() + "'"));      chain.doFilter(request, response);      return;   }   //登录获取Auth   Authentication rememberMeAuth = this.rememberMeServices.autoLogin(request, response);   if (rememberMeAuth != null) {      // Attempt authenticaton via AuthenticationManager      try {      //进行remember-me校验         rememberMeAuth = this.authenticationManager.authenticate(rememberMeAuth);         // Store to SecurityContextHolder         //保存用户实例         SecurityContextHolder.getContext().setAuthentication(rememberMeAuth);         //成功页面跳转         onSuccessfulAuthentication(request, response, rememberMeAuth);         this.logger.debug(LogMessage.of(() -> "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 ex) {         this.logger.debug(LogMessage               .format("SecurityContextHolder not populated with remember-me token, as AuthenticationManager "                     + "rejected Authentication returned by RememberMeServices: '%s'; "                     + "invalidating remember-me token", rememberMeAuth),               ex);         this.rememberMeServices.loginFail(request, response);         //失败页面跳转         onUnsuccessfulAuthentication(request, response, ex);      }   }   chain.doFilter(request, response);}}
@Overridepublic final Authentication autoLogin(HttpServletRequest request, HttpServletResponse response) {   //获取cookie   String rememberMeCookie = extractRememberMeCookie(request);   if (rememberMeCookie == null) {      return null;   }   this.logger.debug("Remember-me cookie detected");   if (rememberMeCookie.length() == 0) {      this.logger.debug("Cookie was empty");      cancelCookie(request, response);      return null;   }   try {       //解码cookie 拿到令牌      String[] cookieTokens = decodeCookie(rememberMeCookie);      //通过令牌获取UserdDetails      UserDetails user = processAutoLoginCookie(cookieTokens, request, response);      this.userDetailsChecker.check(user);      this.logger.debug("Remember-me cookie accepted");      return createSuccessfulAuthentication(request, user);   }   catch (CookieTheftException ex) {      cancelCookie(request, response);      throw ex;   }   catch (UsernameNotFoundException ex) {      this.logger.debug("Remember-me login was valid but corresponding user not found.", ex);   }   catch (InvalidCookieException ex) {      this.logger.debug("Invalid remember-me cookie: " + ex.getMessage());   }   catch (AccountStatusException ex) {      this.logger.debug("Invalid UserDetails: " + ex.getMessage());   }   catch (RememberMeAuthenticationException ex) {      this.logger.debug(ex.getMessage());   }   cancelCookie(request, response);   return null;}

大致整体流程就是如果拿不到实例,则进行remember-me验证,通过autoLogin方法里获取cookie,解析令牌,拿到Auth,最后进行校验。之后剩下的和登陆流程分析的差不多。

以上是“Spring Security如何实现自动登陆功能”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网精选频道!

--结束END--

本文标题: Spring Security如何实现自动登陆功能

本文链接: https://www.lsjlt.com/news/304846.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Spring Security如何实现自动登陆功能
    这篇文章主要介绍Spring Security如何实现自动登陆功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!当我们在登录像QQ邮箱这种大多数的网站,往往在登录按键上会有下次自动登录这个选项,勾选后登录成功,在一段...
    99+
    2023-06-25
  • Spring Security实现自动登陆功能示例
    当我们在登录像QQ邮箱这种大多数的网站,往往在登录按键上会有下次自动登录这个选项,勾选后登录成功,在一段时间内,即便退出浏览器或者服务器重启,再次访问不需要用户输入账号密码进行登录,...
    99+
    2022-11-12
  • Java Web实现自动登陆功能
    Java对自动登陆功能的简单实现,仅用到了servlet和一个filter。第一次写博文,如有不足的地方,敬请指正,谢谢。 登陆界面 <%@ page language="...
    99+
    2022-11-12
  • Spring Security基于散列加密方案实现自动登录功能
    目录前言一. 自动登录简介1. 为什么要自动登录2. 自动登录的实现方案二. 基于散列加密方案实现自动登录1. 配置加密令牌的key2. 配置SecurityConfig类3. 添加...
    99+
    2022-11-12
  • Spring Security如何实现登录验证
    这篇文章主要讲解了“Spring Security如何实现登录验证”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Spring Security如何实现登录验证”吧!一、理...
    99+
    2023-06-26
  • java web如何实现自动登录功能
    这篇文章主要介绍java web如何实现自动登录功能,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!具体内容如下主要思路就是:当用户访问网站的首页时,浏览器端会先检擦浏览器中存在的cookie中是否又登录的用户的用户名...
    99+
    2023-05-31
    java web
  • JavaWeb实现自动登录功能
    本文实例为大家分享了JavaWeb实现自动登录功能的具体代码,供大家参考,具体内容如下 自动登录是通过存储cookie值来实现的。 工程目录如下: login.jsp: <...
    99+
    2022-11-12
  • Spring security如何重写Filter实现json登录
    Spring security 重写Filter实现json登录 在使用SpringSecurity中,大伙都知道默认的登录数据是通过key/value的形式来传递的,默认情况下不支...
    99+
    2022-11-12
  • 如何使用shell实现SSH自动登陆
    小编给大家分享一下如何使用shell实现SSH自动登陆,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!shell脚本基础在编写ssh自动登陆脚本之前,先说一下shell脚本的基础,此基础不是一些语法什么的,网上到处都是,这里...
    99+
    2023-06-09
  • Spring boot security权限管理集成cas单点登录功能的实现
    目录1.Spring boot集成Spring security2.部署CAS server3.配置CAS client挣扎了两周,Spring security的cas终于搞出来了...
    99+
    2022-11-13
  • JSP动态如何实现web网页登陆和注册功能
    这篇文章主要介绍了JSP动态如何实现web网页登陆和注册功能,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。前言 涉及到相关内容如下:数据库安装JSP网页连接数据库注...
    99+
    2023-06-22
  • Spring Security如何实现用户名密码登录
    小编给大家分享一下Spring Security如何实现用户名密码登录,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!引言你在服务端的安全管理使用了 Sp...
    99+
    2023-06-21
  • JS如何实现两周内自动登录功能
    这篇文章将为大家详细讲解有关JS如何实现两周内自动登录功能,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。具体内容如下<!DOCTYPE html> ...
    99+
    2022-10-19
  • spring-boot集成spring-security的oauth2如何实现github登录网站
    这篇文章主要为大家展示了“spring-boot集成spring-security的oauth2如何实现github登录网站”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“spring-boot集...
    99+
    2023-05-30
    spring boot oauth2.0
  • JSP动态实现web网页登陆和注册功能
    目录前言 数据库 JSP网页连接数据库配置mysql文件创建jsp文件,连接数据库,并读取出数据显示在jsp网页中注册和登录的JSP页面新建login.jsp文件新建check.js...
    99+
    2022-11-12
  • 如何实现Putty自动登陆远程Linux主机
    本篇内容主要讲解“如何实现Putty自动登陆远程Linux主机”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何实现Putty自动登陆远程Linux主机”吧!首先登陆到 Linux 主机,具体过...
    99+
    2023-06-09
  • expect如何实现Linux自动登陆远程机器
    本文小编为大家详细介绍“expect如何实现Linux自动登陆远程机器”,内容详细,步骤清晰,细节处理妥当,希望这篇“expect如何实现Linux自动登陆远程机器”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。e...
    99+
    2023-07-04
  • spring boot结合kaptcha怎么实现一个验证码登陆功能
    本篇文章给大家分享的是有关spring boot结合kaptcha怎么实现一个验证码登陆功能,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。引入kaptcha所需要的jar包,我...
    99+
    2023-05-31
    springboot ptc kaptcha
  • 怎么在Java中实现session过期后自动跳转到登陆页功能
    这篇文章主要为大家详细介绍了怎么在Java中实现session过期后自动跳转到登陆页功能,文中示例代码介绍的非常详细,具有一定的参考价值,发现的小伙伴们可以参考一下:Java是什么Java是一门面向对象编程语言,可以编写桌面应用程序、Web...
    99+
    2023-05-30
    java session
  • JavaWeb实现简单的自动登录功能
    本文实例为大家分享了JavaWeb实现简单的自动登录功能的具体代码,供大家参考,具体内容如下 用最近所学的知识点实现自动登录,主要有: 1、Filter过滤器 2、session &...
    99+
    2022-11-12
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作