轻轻松松搞定分布式Token校验

news/2024/5/21 14:58:23/文章来源:https://blog.csdn.net/FUTEROX/article/details/127288002

文章目录

  • 前言
  • token存储
    • Token 存储实体
    • login 业务代码
      • 枚举类修改
    • 存储效果
    • 客户端存储
  • token验证
    • 前端提交
    • 后端校验
      • 自定义注解
      • 切面处理
      • 使用
  • 总结

前言

没想到前天小水了一篇博文,竟然就火了?!!既然如此,那我再来一篇,嘿嘿~
那么今天带来的其实也没啥,就是简简单单的校验,去校验token,然后就好了,但是区别是啥呢,咱们这边有个大冤种就是这个 GateWay。此外这边的全部代码都是在WhiteHolev0.7里面的,可见的。
在这里插入图片描述
由于这个玩意,咱们不好再像以前直接去在拦截器里面去搞事情。而且说实话,请求那么多,如果全部都在GateWay去做的话,我是真的懒得去写那些啥配置了,到时候放行哪些接口都会搞乱。

所以问题背景就是在分布式微服务的场景下,如何去更好地校验token。并且通过我们的token我们可以做到单点登录。

那么这个时候我们就不得不提到我们上篇博文提到的内容了。
SpringBoot轻轻松松搞定用户邮箱登录注册

当然重点是登录模块。

token存储

既然我们要校验,那么我们要做的就是拿到这个token,那么首先要做的就是生成token,然后存储token,咱们上一篇博文已经说的很清楚了,甚至还给出了对应的工具类。我们的流程是这样的:
在这里插入图片描述

那么在这里的话,和先前不一样的是,由于咱们的这个其实是一个多端的,所以的话咱们不仅仅有PC端还有移动端(当然移动端的作者也是我这个大冤种)所以token的话也是要做到多端的。那么这样的话,我们就要对上次做一点改动。

这里的话,和上次不一样的地方有两个。

Token 存储实体

这里新建了一个token的实体,用来存储到redis里面。
在这里插入图片描述


@Data
@AllArgsConstructor
@NoArgsConstructor
public class LoginToken {//这个是我们的存储Redis里面的Tokenprivate String PcLoginToken;private String MobileLoginToken;private String LoginIP;
}

login 业务代码

之后就是我们修改后的代码了。这个也就是和先前做了一点改动,主要是做多端的token嘛。

@Service
public class loginServiceImpl implements LoginService {@AutowiredUserService userService;@AutowiredRedisUtils redisUtils;//为安全期间这里也做一个20防刷@Overridepublic R Login(LoginEntity entity) {String username = entity.getUsername();String password = entity.getPassword();password=password.replaceAll(" ","");if(redisUtils.hasKey(RedisTransKey.getLoginKey(username))){return R.error(BizCodeEnum.OVER_REQUESTS.getCode(),BizCodeEnum.OVER_REQUESTS.getMsg());}redisUtils.set(RedisTransKey.setLoginKey(username),1,20);UserEntity User = userService.getOne(new QueryWrapper<UserEntity>().eq("username", username));if(User!=null){if(SecurityUtils.matchesPassword(password,User.getPassword())){//登录成功,签发token,按照平台类型去签发不同的TokenString token = JwtTokenUtil.generateToken(User);//登录成功后,将userid--->token存redis,便于做登录验证String ipAddr = GetIPAddrUtils.GetIPAddr();if(entity.getType().equals(LoginType.PcType)){LoginToken loginToken = new LoginToken(token,null,ipAddr);redisUtils.set(RedisTransKey.setTokenKey(User.getUserid()+":"+LoginType.PcType),loginToken,7, TimeUnit.DAYS);return Objects.requireNonNull(R.ok(BizCodeEnum.SUCCESSFUL.getMsg()).put(LoginType.PcLoginToken, token)).put("userid",User.getUserid());}else if (entity.getType().equals(LoginType.MobileType)){LoginToken loginToken = new LoginToken(null,token,ipAddr);redisUtils.set(RedisTransKey.setTokenKey(User.getUserid()+":"+LoginType.MobileType),loginToken,7, TimeUnit.DAYS);return Objects.requireNonNull(R.ok(BizCodeEnum.SUCCESSFUL.getMsg()).put(LoginType.PcLoginToken, token)).put("userid",User.getUserid());} else {return R.error(BizCodeEnum.NUNKNOW_LGINTYPE.getCode(),BizCodeEnum.NUNKNOW_LGINTYPE.getMsg());}}else {return R.error(BizCodeEnum.BAD_PUTDATA.getCode(),BizCodeEnum.BAD_PUTDATA.getMsg());}}else {return R.error(BizCodeEnum.NO_SUCHUSER.getCode(),BizCodeEnum.NO_SUCHUSER.getMsg());}}
}

枚举类修改

同样的这里和先前的枚举类有一点不一样,主要是多了一点东西。
在这里插入图片描述

public enum BizCodeEnum {UNKNOW_EXCEPTION(10000,"系统未知异常"),VAILD_EXCEPTION(10001,"参数格式校验失败"),HAS_USERNAME(10002,"已存在该用户"),OVER_REQUESTS(10003,"访问频次过多"),OVER_TIME(10004,"操作超时"),BAD_DOING(10005,"疑似恶意操作"),BAD_EMAILCODE_VERIFY(10007,"邮箱验证码错误"),REPARATION_GO(10008,"请重新操作"),NO_SUCHUSER(10009,"该用户不存在"),BAD_PUTDATA(10010,"信息提交错误,请重新检查"),NOT_LOGIN(10011,"用户未登录"),BAD_LOGIN_PARAMS(10012,"请求异常!触发5次以上账号将保护性封禁"),NUNKNOW_LGINTYPE(10013,"平台识别异常"),BAD_TOKEN(10014,"token校验失败"),SUCCESSFUL(200,"successful");private int code;private String msg;BizCodeEnum(int code,String msg){this.code = code;this.msg = msg;}public int getCode() {return code;}public String getMsg() {return msg;}
}

当然同样的,多的东西还有几个异常类,这个其实就是继承了Exception。

/*** 校验用户登录时,参数不对的情况,此时可能是恶意爬虫* */
public class BadLoginParamsException extends Exception{public BadLoginParamsException(){}public BadLoginParamsException(String message){super(message);}}

public class BadLoginTokenException extends Exception{public BadLoginTokenException(){}public BadLoginTokenException(String message){super(message);}
}
public class NotLoginException extends Exception{public NotLoginException(){}public NotLoginException(String message){super(message);}
}

其他的倒还是和先前的保持一致。

存储效果

那么到此我们在登录部分完成了对token的存储,但是这个是在服务端,现在这个玩意已经存到了咱们的redis里面:
在这里插入图片描述

客户端存储

现在我们服务端已经存储好了,那么接下来就是要在客户端进行存储。这个也好办,我们直接来看到完整的用户登录代码就知道了。

<template><div><el-form :model="formLogin" :rules="rules" ref="ruleForm" label-width="0px" ><el-form-item prop="username"><el-input v-model="formLogin.username" placeholder="账号"><i slot="prepend" class="el-icon-s-custom"/></el-input></el-form-item><el-form-item prop="password"><el-input type="password" placeholder="密码" v-model="formLogin.password"><i slot="prepend" class="el-icon-lock"/></el-input></el-form-item><el-form-item prop="code"><el-row :span="24"><el-col :span="12"><el-input v-model="formLogin.code" auto-complete="off"  placeholder="请输入验证码" size=""></el-input></el-col><el-col :span="12"><div class="login-code" @click="refreshCode"><!--验证码组件--><s-identify :identifyCode="identifyCode"></s-identify></div></el-col></el-row></el-form-item><el-form-item><div class="login-btn"><el-button type="primary" @click="submitForm()" style="margin-left: auto;width: 35%">登录</el-button><el-button type="primary" @click="goRegister" style="margin-left: 27%;width: 35%" >注册</el-button></div></el-form-item></el-form></div>
</template><script>
import SIdentify from "../../components/SIdentify/SIdentify";
export default {name: "loginbyUserName",components: { SIdentify },data() {return{formLogin: {username: "",password: "",code: ""},identifyCodes: '1234567890abcdefjhijklinopqrsduvwxyz',//随机串内容identifyCode: '',// 校验rules: {username:[{ required: true, message: "请输入用户名", trigger: "blur" }],password: [{ required: true, message: "请输入密码(区分大小写)", trigger: "blur" }],code: [{ required: true, message: "请输入验证码", trigger: "blur" }]}}},mounted () {// 初始化验证码this.identifyCode = ''this.makeCode(this.identifyCodes, 4)},methods:{refreshCode () {this.identifyCode = ''this.makeCode(this.identifyCodes, 4)},makeCode (o, l) {for (let i = 0; i < l; i++) {this.identifyCode += this.identifyCodes[this.randomNum(0, this.identifyCodes.length)]}},randomNum (min, max) {return Math.floor(Math.random() * (max - min) + min)},submitForm(){if (this.formLogin.code.toLowerCase() !== this.identifyCode.toLowerCase()) {this.$message.error('请填写正确验证码')this.refreshCode()}else {//这边后面做一个提交,服务器验证,通过之后获得tokenthis.axios({url: "/user/user/login",method: 'post',data:{"username":this.formLogin.username,"password":this.formLogin.password,"type": "PcType",}}).then((res)=>{res = res.dataif (res.code===10001){alert("请将对应信息填写完整!")}else if(res.code===0){alert("登录成功")localStorage.setExpire("LoginToken",res.PcLoginToken,this.OverTime)localStorage.setExpire("userid",res.userid,this.OverTime)this.$router.push({ path: '/userinfo', query: {'userid':res.userid} });}else {alert(res.msg);}})}},goRegister(){this.$router.push("/register")}},
}
</script><style scoped>
</style>

这里的话,咱们对localStorage做了一点优化:
这个代码是在main.js直接搞的。

Storage.prototype.setExpire=(key, value, expire) =>{let obj={data:value,time:Date.now(),expire:expire};localStorage.setItem(key,JSON.stringify(obj));
}
//Storage优化
Storage.prototype.getExpire= key =>{let val =localStorage.getItem(key);if(!val){return val;}val =JSON.parse(val);if(Date.now()-val.time>val.expire){localStorage.removeItem(key);return null;}return val.data;
}

这个this.OverTime 就是一个全局变量,就是7天过期的意思。

token验证

前面咱们说完了这个存储,那么现在的话咱们就是验证服务了。首先我们来看到什么地方需要验证。
我们拿这个为例子:
在这里插入图片描述

主页的话,都是get请求,没啥技术含量,不过我不介意再水一篇博客~。那么就是咱们这个页面需要。

那么在这里的话我先说一下执行流程,这样的话咱们完整的案例就起来了:
在这里插入图片描述

前端提交

那么现在咱们来看看前端的代码:

<script>
export default {name: "myspace",data() {return {}},created() {//先对token再进行验证let loginToken = localStorage.getExpire("LoginToken");let userid = localStorage.getExpire("userid");//这个只有用户自己才能进入,自己只能进入自己对应的MySpaceif(loginToken==null && userid==null){alert("检测到您未登录,请先登录")this.$router.push({path: "/login"});}else {//发送token验证token是否正常,否则一样不给过this.axios({url: "/user/user/space/isLogin",method: 'get',headers: {"userid": userid,"loginType": "PcType","loginToken": loginToken,},params: {'userid': userid,}}).then((res)=>{res = res.data;if (!(res.code === 0)) {alert(res.msg)this.$router.push({path: "/login"});}}).catch((err)=>{alert("未知异常,请重新登录")this.$router.push({path: "/login"});});}}
}
</script>

前面的那些玩意没啥用,咱们直接看到这个实际执行的代码。

后端校验

ok,现在咱们可以来聊聊这个后端的校验了,这个还是很重要的,也是咱们今天的主角。
那么在开始的时候咱们说了这个使用拦截器的方案并不是可行的,而且在后面可能我们还需要在业务处理的时候拿到token去解析里面的东西,完成一些处理,到时候在拦截器的时候也不好处理。而且重点是并不是所有的接口都要的,但是也不是少部分的接口不要,这TM就尴尬了,那么如何破局。那么此时我们就需要定位到每一个具体的方法上面,那么问题不就解决了,这个咋搞,诶嘿,搞个切面+注解不就完了。

自定义注解

先定义一个注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface NeedLogin {String value() default "";
}

这个注解我放在了common组件下:
在这里插入图片描述

切面处理

那么之后就是咱们的切面了,我们刚刚定义的异常处理类都是在这个切面上处理的。
在这里插入图片描述

public class VerificationAspect {@AutowiredRedisUtils redisUtils;@Pointcut("@annotation(com.huterox.common.holeAnnotation.NeedLogin)")public void verification() {}/*** 环绕通知 @Around ,当然也可以使用 @Before (前置通知)  @After (后置通知)就算了* @param proceedingJoinPoint* @return* 我们这里再直接抛出异常,反正有那个谁统一异常类*/@Around("verification()")public Object verification(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();ServletRequestAttributes servletRequestAttributes = (ServletRequestAttributes) requestAttributes;assert servletRequestAttributes != null;HttpServletRequest request = servletRequestAttributes.getRequest();//分登录的设备进行验证String loginType = request.getHeader("loginType");String userid = request.getHeader("userid");String tokenUser = request.getHeader("loginToken");String tokenKey = RedisTransKey.getTokenKey(userid + ":" + loginType);if(tokenUser==null || userid==null || loginType==null){throw new BadLoginParamsException();}if(redisUtils.hasKey(tokenKey)){if(loginType.equals(LoginType.PcType)){Object o = redisUtils.get(tokenKey);LoginToken loginToken = JSON.parseObject(o.toString(), LoginToken.class);if(!loginToken.getPcLoginToken().equals(tokenUser)){throw new BadLoginTokenException();}}else if (loginType.equals(LoginType.MobileType)){Object o = redisUtils.get(tokenKey);LoginToken loginToken = JSON.parseObject(o.toString(), LoginToken.class);if(!loginToken.getMobileLoginToken().equals(tokenUser)){throw new BadLoginTokenException();}}}else {throw new NotLoginException();}return proceedingJoinPoint.proceed();}
}

使用

那么接下来就是使用了。我们来看到这个:
在这里插入图片描述

这个是我们的controller,作用就是用来检验这个用户本地的token对不对的,那么实现的服务类啥也没有:
在这里插入图片描述

之后我们来看到咱们的一个效果:
在这里插入图片描述
可以看到在进入页面的时候,钩子函数会请求咱们的这个接口,然后的话,咱们通过这个接口的话可以看到验证的效果。这里验证通过了。

总结

让我康康这篇文章的效果咋样,If it works well, I’ll take out my development log directly and go to Bling Bling your eyes.!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_22417.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

第7章 单行函数

1.函数的理解 *函数可以把我们经常使用的代码封装起来&#xff0c;需要的时候直接调用即可。这样既提高了代码效率&#xff0c;又提高了可维护性。在SQL中我们也可以使用函数对检索出来的数据进行函数操作。使用这些函数&#xff0c;可以极大地提高用户对数据库的管理效率。 …

微信小程序|基于小程序实现打卡功能

文章目录一、文章前言二、开发流程及准备三、开发步骤一、文章前言 此文主要在小程序内实现打卡功能&#xff0c;可根据用户位置与公司设定的打卡范围实时判断打卡场景。 二、开发流程及准备 2.1、注册微信公众平台账号。 2.2、准备腾讯地图用户Key。 三、开发步骤 3.1、访问…

【面试题常考!!!】JZ39 数组中出现次数超过一半的数字【五种方法解决】

欢迎观看我的博客&#xff0c;如有问题交流&#xff0c;欢迎评论区留言&#xff0c;一定尽快回复&#xff01;&#xff08;大家可以去看我的专栏&#xff0c;是所有文章的目录&#xff09; 字体风格&#xff1a; 红色文字表示&#xff1a;重难点 蓝色文字表示&#xff1a;思路以…

神经网络模型数据处理,神经网络模型参数辨识

1、有哪些深度神经网络模型&#xff1f; 目前经常使用的深度神经网络模型主要有卷积神经网络(CNN) 、递归神经网络(RNN)、深信度网络(DBN) 、深度自动编码器(AutoEncoder) 和生成对抗网络(GAN) 等。 递归神经网络实际.上包含了两种神经网络。一种是循环神经网络(Recurrent Neu…

STM32F4单片机读取AT24c02

​STM32F4是由ST&#xff08;意法半导体&#xff09;开发的一种高性能微控制器系列。其采用了90nm的NVM工艺和ART技术&#xff08;自适应实时存储加速器&#xff0c;Adaptive Real-Time MemoryAccelerator™&#xff09; AT24C02是Atmel公司出品的一个2K位串行CMOS E2PROM&…

【k8s】五、Pod生命周期(一)

目录 前言 Pod生命周期 Pod 相位 状态值 挂起&#xff08;Pending&#xff09; 运行中&#xff08;Running&#xff09; 成功&#xff08;Succeeded&#xff09; 失败&#xff08;Failed&#xff09; 未知&#xff08;Unknown&#xff09; Init Containers Init Cont…

pc端引擎颠覆电脑兼容性

张小龙曾在讲座上阐述小程序理念的精髓&#xff0c;小程序承载着张小龙及微信团队对未来程序形态的一种见解&#xff0c;总结为五个字&#xff1a;所见即所得。原文如下&#xff1a; 它是一种真正的所见即所得的形态&#xff0c;我说的所见即所得不同于在PC时代&#xff0c;我…

组合模式+桥接模式

目录 组合模式 定义&#xff1a; 业务实现例子&#xff1a; 桥接模式 JDBC中的桥接模式 组合模式 定义&#xff1a; 将对象组合通过树形结构进行展示&#xff0c;使得用户——>不管对单个对象or组合对象的使用具有一致性 可以理解为部分-整体模式——>简单来说就…

深度学习环境搭建

(1) 安装 Anaconda :建立 Python 应用环境 安装成功界面如下:(2) Visual Studio Code: 建立代码编辑环境 1.安装Python扩展2.选择合适的Python解释器 3.安装下列应用扩展:codeRunner : 快速运行程序 Jupyter : 交互式运行程序 Pylance : 高效代码提示 安装完成如图所示:4.创…

Linux基础组件之muduo日志库分析

muduo日志库分析异步日志机制双缓存机制前台日志写入栈后台日志(落盘)写入栈使用示例总结后言异步日志机制 #mermaid-svg-nrIugWYiOaAGFTWH {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-nrIugWYiOaAGFTWH .error-…

如何做架构规划

文章目录架构师的职责WhyWhatHow架构活动生命周期环境搭建目标确认可行性探索架构规划统一语义需求确认任务边界划分确认规划完整性项目启动阶段性价值交付复盘经历过的典型案例参考架构师的职责 Why 互联网架构活动的挑战较多&#xff0c;如&#xff1a; 反射式的研发行为。…

Scratch软件编程等级考试四级——20200913

Scratch软件编程等级考试四级——20200913理论单选题判断题实操奇偶之和创意画图数字之和用逗号分隔列表数字反转理论 单选题 1、执行下面程序&#xff0c;输入4和7后&#xff0c;角色说出的内容是&#xff1f;&#xff08;&#xff09; A、4&#xff0c;7 B、7,7 C、7,4 D、…

为什么会发生云中断?如何防范?

IT 越依赖云服务&#xff0c;用户就越有可能因云中断而遭受停机和收入损失。由于云中断事件的发生&#xff0c;超过 60% 的使用公共云的组织在 2022 年报告了损失&#xff0c;因此云中断并不是公司不太可能面临的异常事件。 但是中断是否足以成为永远离开云的理由?还是应该坚持…

《安富莱嵌入式周报》第286期:8bit浮点数规范,VxWorks火星探测器故障原因修复,Matter V1.0智能家居规范,Wireshark 4.0发布

往期周报汇总地址&#xff1a;嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 目录 视频版&#xff1a; 1、SIA全球半导体行业协会统计显示全球芯片市场增长放缓&#xff0c;中国市场下跌10% …

程序员如何高效准备简历和面试03:诊断:简历为什么被忽视?

你好&#xff0c;欢迎学习课时3&#xff0c;我是你的职场导师吴文娟。 这节课主要为后面教你写简历做个铺垫&#xff0c;学习内容只有2个字&#xff1a;挑错。一个大家比较喜欢的事。我们来敲黑板看一些反面典型&#xff0c;案例都是我截取之前诊断过的简历&#xff0c;讲一讲为…

Mac电脑图片后期处理Lightroom Classic 2022(lrc2022)

Lightroom Classic 2022具有非常强大的图像处理功能&#xff0c;甚至对照片的一些修饰也可以完成&#xff0c;例如去除不要的物体、校正照片和增强照片颜色等。Lightroom Classic 2022 Mac版为用户提供了各种满足优秀摄影效果所需的编辑工具。让您能够轻松提亮颜色、使灰暗的摄…

C++ Builder XE TChart动态添加N个线条TLineSeries变化

// LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq (double)litmp.QuadPart;// 获得计数器的时钟频率 QueryPerformanceCounter(&litmp)…

STM32:外部中断控制旋转编码器并计次

1.主函数(main.c)代码部分&#xff1a; #include "stm32f10x.h" // Device header #include "Delay.h" #include "OLED.h" #include "Encoder.h" int16_t Num; int main(void) { OLED_Init(); OLED_Sh…

第十四届蓝桥杯备赛模板题——蓝桥部队 (带权并查集)

目录1.蓝桥部队1. 问题描述2.输入格式3.输入样例4.样例答案5.原题连接2.解题思路3.Ac_code1.蓝桥部队 1. 问题描述 小明是蓝桥部队的长官&#xff0c;他的班上有 NNN 名军人和 111 名军师。 这天&#xff0c;NNN 名军人在操场上站成一排&#xff0c;起初编号为 iii 的军人站…

JPA EntityManager 获取关联对象

今天尝试了几种方式&#xff0c;来获取关联对象。 关联对象&#xff0c;使用的 OneToMany(fetchFetchType.EAGER) 下面给一下总结&#xff1a; 一 Example 毫无疑问&#xff0c;很有信心&#xff0c;Example可以关联到对象。 事实也是这样。 但是Example好像只有and关系…