• 使用 Sa-Token 实现 [记住我] 模式登录、七天免登录


    一、需求分析

    如图所示,一般网站的登录界面都会有一个 [记住我] 按钮,当你勾选它登录后,即使你关闭浏览器再次打开网站,也依然会处于登录状态,无须重复验证密码:

    ../static/login-view.png

    本文将详细介绍在 Sa-Token中,如何做到以下登录模式:

    • 记住我登录:登录后关闭浏览器,再次打开网站登录状态依然有效,无需重复登录。
    • 仅本次有效登录:登录后关闭浏览器,再次打开网站登录状态将失效,需要再次登录。
    • 七天免登录:为登录状态设定一个详细的有效期,在这个期限内无需重复登录,过了期限后需要再次登录。

    Sa-Token 是一个轻量级 java 权限认证框架,主要解决登录认证、权限认证、单点登录、OAuth2、微服务网关鉴权 等一系列权限相关问题。

    首先在项目中引入 Sa-Token 依赖:

    
    <dependency>
        <groupId>cn.dev33groupId>
        <artifactId>sa-token-spring-boot-starterartifactId>
        <version>1.34.0version>
    dependency>
    

    注:如果你使用的是 SpringBoot 3.x,只需要将 sa-token-spring-boot-starter 修改为 sa-token-spring-boot3-starter 即可。

    二、在 Sa-Token 中实现记住我功能

    Sa-Token的登录授权,默认就是[记住我]模式,为了实现[非记住我]模式,你需要在登录时如下设置:

    // 设置登录账号id为10001,第二个参数指定是否为[记住我],当此值为false后,关闭浏览器后再次打开需要重新登录
    StpUtil.login(10001, false);
    

    那么,Sa-Token实现[记住我]的具体原理是?

    三、实现原理

    Cookie作为浏览器提供的默认会话跟踪机制,其生命周期有两种形式,分别是:

    • 临时Cookie:有效期为本次会话,只要关闭浏览器窗口,Cookie就会消失。
    • 持久Cookie:有效期为一个具体的时间,在时间未到期之前,即使用户关闭了浏览器Cookie也不会消失。

    利用Cookie的此特性,我们便可以轻松实现 [记住我] 模式:

    • 勾选 [记住我] 按钮时:调用StpUtil.login(10001, true),在浏览器写入一个持久Cookie储存 Token,此时用户即使重启浏览器 Token 依然有效。
    • 不勾选 [记住我] 按钮时:调用StpUtil.login(10001, false),在浏览器写入一个临时Cookie储存 Token,此时用户在重启浏览器后 Token 便会消失,导致会话失效。

    动态演示图:

    记住我登录

    四、前后端分离模式下如何实现[记住我]?

    此时机智的你😏很快发现一个问题,Cookie虽好,却无法在前后端分离环境下使用,那是不是代表上述方案在APP、小程序等环境中无效?

    准确的讲,答案是肯定的,任何基于Cookie的认证方案在前后端分离环境下都会失效(原因在于这些客户端默认没有实现Cookie功能),不过好在,这些客户端一般都提供了替代方案,
    唯一遗憾的是,此场景中token的生命周期需要我们在前端手动控制:

    以经典跨端框架 uni-app 为例,我们可以使用如下方式达到同样的效果:

    // 使用本地存储保存token,达到 [持久Cookie] 的效果
    uni.setStorageSync("satoken", "xxxx-xxxx-xxxx-xxxx-xxx");
    
    // 使用globalData保存token,达到 [临时Cookie] 的效果
    getApp().globalData.satoken = "xxxx-xxxx-xxxx-xxxx-xxx";
    

    如果你决定在PC浏览器环境下进行前后端分离模式开发,那么更加简单:

    // 使用 localStorage 保存token,达到 [持久Cookie] 的效果
    localStorage.setItem("satoken", "xxxx-xxxx-xxxx-xxxx-xxx");
    
    // 使用 sessionStorage 保存token,达到 [临时Cookie] 的效果
    sessionStorage.setItem("satoken", "xxxx-xxxx-xxxx-xxxx-xxx");
    

    Remember me, it's too easy!

    五、登录时指定 Token 有效期

    登录时不仅可以指定是否为[记住我]模式,还可以指定一个特定的时间作为 Token 有效时长,如下示例:

    // 示例1:
    // 指定token有效期(单位: 秒),如下所示token七天有效
    StpUtil.login(10001, new SaLoginModel().setTimeout(60 * 60 * 24 * 7));
    
    // ----------------------- 示例2:所有参数
    // `SaLoginModel`为登录参数Model,其有诸多参数决定登录时的各种逻辑,例如:
    StpUtil.login(10001, new SaLoginModel()
    			.setDevice("PC")                 // 此次登录的客户端设备类型, 用于[同端互斥登录]时指定此次登录的设备类型
    			.setIsLastingCookie(true)        // 是否为持久Cookie(临时Cookie在浏览器关闭时会自动删除,持久Cookie在重新打开后依然存在)
    			.setTimeout(60 * 60 * 24 * 7)    // 指定此次登录token的有效期, 单位:秒 (如未指定,自动取全局配置的 timeout 值)
    			.setToken("xxxx-xxxx-xxxx-xxxx") // 预定此次登录的生成的Token 
    			.setIsWriteHeader(false)         // 是否在登录后将 Token 写入到响应头
    			);
    

    注:如果在登录时未指定 new SaLoginModel().setTimeout(604800) 那么框架将采用全局配置的 sa-token.timeout 值作为 Token 的有效期。

    六、不同登录策略的代码对比

    以下是三种登录策略的代码差异:

    package com.pj.cases.up;
    
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import cn.dev33.satoken.stp.StpUtil;
    import cn.dev33.satoken.util.SaResult;
    
    /**
     * Sa-Token 记住我模式登录 
     * 
     * @author kong
     * @since 2022-10-17 
     */
    @RestController
    @RequestMapping("/RememberMe/")
    public class RememberMeController {
    
    	// 记住我登录    ---- http://localhost:8081/RememberMe/doLogin?name=zhang&pwd=123456
    	@RequestMapping("doLogin")
    	public SaResult doLogin(String name, String pwd) {
    		if("zhang".equals(name) && "123456".equals(pwd)) {
    			StpUtil.login(10001, true);
    			return SaResult.ok("登录成功");
    		}
    		return SaResult.error("登录失败");
    	}
    	
    	// 不记住我登录    ---- http://localhost:8081/RememberMe/doLogin2?name=zhang&pwd=123456
    	@RequestMapping("doLogin2")
    	public SaResult doLogin2(String name, String pwd) {
    		if("zhang".equals(name) && "123456".equals(pwd)) {
    			StpUtil.login(10001, false);
    			return SaResult.ok("登录成功");
    		}
    		return SaResult.error("登录失败");
    	}
    
    	// 七天免登录    ---- http://localhost:8081/RememberMe/doLogin3?name=zhang&pwd=123456
    	@RequestMapping("doLogin3")
    	public SaResult doLogin3(String name, String pwd) {
    		if("zhang".equals(name) && "123456".equals(pwd)) {
    			StpUtil.login(10001, 60 * 60 * 24 * 7);
    			return SaResult.ok("登录成功");
    		}
    		return SaResult.error("登录失败");
    	}
    	
    }
    

    可依次访问注释中提供的测试链接,观察不同登录策略带来的会话有效期差异。


    参考资料

  • 相关阅读:
    Windows下Vscode找不到头文件的分析
    怎么保护公司文件安全
    股权项目披露:扬州国扬电子有限公司6.2664%股权转让
    Python实现全自动输入文本
    数据挖掘实验(Apriori,fpgrowth)
    unity 骨骼物理 头发 布料模拟
    2023年深圳市专精特新企业申报详解与答疑汇总!
    Linux系统--------内核参数调优、一键安装nginx、tomcat调优
    MySQL查询进阶——从函数到表连接的使用你还记得吗
    Golang模块级私有包(internal package mechanism)详解
  • 原文地址:https://www.cnblogs.com/shengzhang/p/17490173.html