Golang JWT 认证 (三)-添加token自动刷新机制

news/2024/5/2 10:27:55/文章来源:https://blog.csdn.net/LeoForBest/article/details/126669714

文章目录

    • 一: 实现原理
      • 1. 后端中间件改进
      • 2. 前端改进
      • 3. 过期后点击请求测试
    • 二: 完整代码
      • 后端
      • 前端
    • 三: 其他思路

上一个Demo中,token一旦过期无法刷新需要重新登录,因此需要某种机制来自动更新token

一: 实现原理

1. 后端中间件改进

读取token中过期时间
token过期时间-最大截止刷新时间>当前时间?
返回结果添加一个http头new-token=新生成的token
End
//gin jwt 认证中间件
func AuthRequired() gin.HandlerFunc {return func(ctx *gin.Context) {tokenString := strings.TrimPrefix(ctx.GetHeader("Authorization"), "Bearer ")token, err := jwt.ParseWithClaims(tokenString, &customClaims{}, func(t *jwt.Token) (interface{}, error) { return jwtKey, nil })if err != nil {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": fmt.Sprintf("access token parse error: %v", err)})return}if claims, ok := token.Claims.(*customClaims); ok && token.Valid {if !claims.VerifyExpiresAt(time.Now(), false) {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": "access token expired"})return}// *******************************新增部分***********************************// 即将超过过期时间,则添加一个http header `new-token` 给前端更新if t := claims.ExpiresAt.Time.Add(-time.Minute * TOKEN_MAX_REMAINING_MINUTE); t.Before(time.Now()) {claims := customClaims{Username: claims.Username,IsAdmin:  claims.Username == "admin",RegisteredClaims: jwt.RegisteredClaims{ExpiresAt: &jwt.NumericDate{Time: time.Now().Add(TOKEN_MAX_EXPIRE_HOUR * time.Hour)},},}// *******************************新增部分结束*******************************token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)tokenString, _ := token.SignedString(jwtKey)ctx.Header("new-token", tokenString)}ctx.Set("claims", claims)} else {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": fmt.Sprintf("Claims parse error: %v", err)})return}ctx.Next()}
}

2. 前端改进

请求API
返回http头是否包含new-token?
更新localStorage中token
End

3. 过期后点击请求测试

在这里插入图片描述

二: 完整代码

后端

package mainimport ("fmt""log""net/http""strings""time""github.com/gin-gonic/gin""github.com/golang-jwt/jwt/v4"
)var jwtKey []byte = []byte("secret")const (TOKEN_MAX_EXPIRE_HOUR      = 1  // token最长有效期TOKEN_MAX_REMAINING_MINUTE = 15 // token还有多久过期就返回新token
)type customClaims struct {Username string `json:"username"`IsAdmin  bool   `json:"IsAdmin"`jwt.RegisteredClaims
}//gin jwt 认证中间件
func AuthRequired() gin.HandlerFunc {return func(ctx *gin.Context) {tokenString := strings.TrimPrefix(ctx.GetHeader("Authorization"), "Bearer ")token, err := jwt.ParseWithClaims(tokenString, &customClaims{}, func(t *jwt.Token) (interface{}, error) { return jwtKey, nil })if err != nil {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": fmt.Sprintf("access token parse error: %v", err)})return}if claims, ok := token.Claims.(*customClaims); ok && token.Valid {if !claims.VerifyExpiresAt(time.Now(), false) {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": "access token expired"})return}// 即将超过过期时间,则添加一个http header `new-token` 给前端更新if t := claims.ExpiresAt.Time.Add(-time.Minute * TOKEN_MAX_REMAINING_MINUTE); t.Before(time.Now()) {claims := customClaims{Username: claims.Username,IsAdmin:  claims.Username == "admin",RegisteredClaims: jwt.RegisteredClaims{ExpiresAt: &jwt.NumericDate{Time: time.Now().Add(TOKEN_MAX_EXPIRE_HOUR * time.Hour)},},}token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)tokenString, _ := token.SignedString(jwtKey)ctx.Header("new-token", tokenString)}ctx.Set("claims", claims)} else {ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"code": -1, "msg": fmt.Sprintf("Claims parse error: %v", err)})return}ctx.Next()}
}type loginRequest struct {Username string `json:"username"`Password string `json:"password"`
}func main() {r := gin.Default()r.POST("/auth/login", func(ctx *gin.Context) {var req loginRequestctx.BindJSON(&req)if req.Username != req.Password {ctx.JSON(http.StatusOK, gin.H{"code": -1, "msg": "incorrect username or password"})return}log.Printf("login user " + req.Username)claims := customClaims{Username: req.Username,IsAdmin:  req.Username == "admin",RegisteredClaims: jwt.RegisteredClaims{ExpiresAt: &jwt.NumericDate{Time: time.Now().Add(TOKEN_MAX_EXPIRE_HOUR * time.Hour)},},}token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)if tokenString, err := token.SignedString(jwtKey); err != nil {ctx.JSON(http.StatusOK, gin.H{"code": -1, "msg": "generate access token failed: " + err.Error()})} else {ctx.JSON(http.StatusOK, gin.H{"code": 0, "msg": "", "data": tokenString})}})api := r.Group("/api")api.Use(AuthRequired())api.GET("/test", func(ctx *gin.Context) {claims := ctx.MustGet("claims").(*customClaims)ctx.JSON(http.StatusOK, gin.H{"code": 0, "data": fmt.Sprintf("current user: %v , is admin: %v", claims.Username, claims.IsAdmin)})})r.Run(":8080")
}

前端

<template><el-config-provider namespace="ep"><el-menu class="el-menu-demo" mode="horizontal"><el-menu-item index="1">Vue JWT Demo</el-menu-item><el-menu-item index="2">当前用户: {{ username }}</el-menu-item></el-menu><!-- <img alt="Vue logo" class="element-plus-logo" src="./assets/logo.png" /> --><el-row style="margin-top: 2rem"><el-col :span="8"></el-col><el-col :span="8"><template v-if="username === ''"><el-form :model="form"><el-form-item label="用户"><el-input v-model="form.username" placeholder="username" /></el-form-item><el-form-item label="密码"><el-input v-model="form.password" placeholder="password" /></el-form-item><el-form-item><el-button type="primary" style="width: 100%" @click="onLogin">获取Token</el-button></el-form-item></el-form></template><template v-else><template v-if="result !== ''"><h1>请求成功</h1><h2>{{ result }}</h2></template><el-button type="primary" @click="onAPI">请求API</el-button><el-button @click="onLogout">退出登陆</el-button></template></el-col><el-col :span="8"></el-col></el-row><!-- <HelloWorld msg="Hello Vue 3.0 + Element Plus + Vite" /> --></el-config-provider>
</template><script setup lang="ts">
import { reactive, ref } from 'vue'
import axios from 'axios'
import { ElMessage } from 'element-plus';const form = reactive({username: '',password: '',
})const username = ref(localStorage.getItem('username') || '')
const result = ref('')const onLogin = () => {axios.post('/auth/login', { username: form.username, password: form.password }).then(response => {if (response.data.code !== 0) {ElMessage.warning(`登陆失败: ${response.data.msg}`)} else {localStorage.setItem('token', response.data.data);localStorage.setItem('username', form.username);username.value = form.username;ElMessage.info(`${form.username}登陆成功!`)}}).catch(err => {console.log(err)ElMessage.error(err)})
}const onAPI = () => {axios.get('/api/test', { headers: { 'Authorization': 'Bearer ' + localStorage.getItem('token') } }).then(response => {if (response.headers['new-token']) {localStorage.setItem('token', response.headers['new-token']);}if (response.data.code !== 0) {ElMessage.warning(`获取: ${response.data.msg}`)} else {result.value = response.data.dataElMessage.info(`请求成功!`)}}).catch(err => {console.log(err)ElMessage.error(err)})
}const onLogout = () => {localStorage.removeItem('username')localStorage.removeItem('token')username.value = ''
}
</script><style>
#app {text-align: center;color: var(--ep-text-color-primary);
}/* 
.element-plus-logo {width: 50%;
} */
</style>

三: 其他思路

  • 后端添加一个refresh_token路由,由客户端判断过期时间,请求该路由
	api.GET("/refresh-token", func(ctx *gin.Context) {claims := ctx.MustGet("claims").(*customClaims)newClaims := customClaims{Username: claims.Username,IsAdmin:  claims.Username == "admin",RegisteredClaims: jwt.RegisteredClaims{ExpiresAt: &jwt.NumericDate{Time: time.Now().Add(TOKEN_MAX_EXPIRE_HOUR * time.Hour)},},}token := jwt.NewWithClaims(jwt.SigningMethodHS256, newClaims)tokenString, _ := token.SignedString(jwtKey)ctx.JSON(http.StatusOK, gin.H{"code": 0, "msg": "", "data": tokenString})})
  • 前段存储token时,在localStorage中再添加一个expire时间字段,每次请求API时判断是否快过期,并决定是否更新为新的token和expire时间

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

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

相关文章

MybatisPlus笔记

MyBatis-PlusMyBatis-Plus概述 需要基础:学习过Spring、SpringMVC、Mybatis 为什么要学习它呢?MyBatisPlus可以节省我们大量的工作时间,所有的CRUD代码都可以自动化完成! JPA、tk-mapper、MyBatisPlus 1、简介 是什么? Mybatis本来就是简化JDBC操作的! 官网:MyBatis-Plu…

润和软件携OpenHarmony亮相全国首场华为云云商店·星品推介会

8月24日&#xff0c;全国首场华为云云商店星品推介会——硬件云服务&#xff08;深圳站&#xff09;在深圳天安云谷成功举办。本次会议以“端云协同 创新生态”为主题&#xff0c;对合作伙伴和客户的最新成果和实践经验进行展示分享&#xff0c;润和软件受邀参会并发表主题演讲…

猿创征文 |【Ant Design Pro】使用ant design pro做为你的开发模板(一)拉取项目

关于我&#xff1a;明月&#xff0c;从业前端开发&#xff0c;会Java&#xff0c;会前端&#xff0c;会产品&#xff0c;会测试&#xff0c;会大客户销售&#xff0c;有过职业规划经验&#xff0c;欢迎各位私信聊天。目标是创业实现上班自由。梦想明月天涯。 关于社区&#xff…

spring boot 服务使用过程常见bug 解决

spring boot服务使用过程常见bug 今天开始持续汇总&#xff1a; 1、【Springboot端口号占用】Web server failed to start. Port xxxxx was already in use. 检查此端口号&#xff1a; tasklist|findstr "50010"cmd命令查看端口号占用情况&#xff0c;例如查看端口5…

spring+aliyunONS

1.阿里云ONS是什么&#xff1f; 消息队列RocketMQ版&#xff08;原名开放消息服务&#xff0c;简称ONS&#xff09;是阿里云基于Apache RocketMQ构建的低延迟、高并发、高可用、高可靠的分布式消息中间件。 2.阿里云RocketMq免费试用一个月网址 阿里云试用中心_云服务器试用_企…

[iOS]-网络请求总结

目录&#xff1a;参考的博客&#xff1a;最原始的网络下载 --- NSData NSURL方式NSURLConnection 和 NSURLSessionGET请求下载完成的事件采用block形式下载完成的事件采用delegate形式POST请求GET和POST操作的区别使用情况使用POST方法使用GET方法HTTP与HTTPSNSURLSessionConf…

APS智能排产助力印染行业进行精细化管理

根据国家统计局数据&#xff0c;2022年1-6月&#xff0c;规模以上印染企业营业收入1500.66亿元&#xff0c;同比增长11.10%&#xff1b;实现利润总额57.18亿元&#xff0c;同比增长13.79%&#xff1b;成本费用利润率4.04%&#xff0c;同比提高0.10个百分点&#xff1b;销售利润…

大学公众号题库API 网课查题题库接口API接口

大学公众号题库API 网课查题题库接口API接口 本平台优点&#xff1a; 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a; 题库&#xff1a;题库后台http://ch…

Problem P04. [算法课分治] 找到 k 个最小数

先sort排序,在输出最小的k个数。#include<iostream> #include<bits/stdc++.h> #include<cstdio>using namespace std;int n, k; int arr[10005];int main() {scanf("%d %d", &n, &k);for (int i = 0; i < n; i++){scanf("%d"…

【Hive】各种join连接用法

目录 一、简介 二、创建数据 1、数据概览 2、创建hive表并插入数据 三、join连接测试 1、join(inner join) 2、left join(left outer join) 3、right join(right outer join) 4、full join(full outer join) 5、left semi join 6、map side join 四、join 和 left …

定时任务cron

原文链接 1 格式 {秒数} {分钟} {小时} {日期} {月份} {星期} {年份(可为空)} 2 用法 "30 * * * * ? " 每半分钟触发任务"30 10 * * * ? " 每小时的10分30秒触发任务"30 10 1 * * ? " 每天1点10分30秒触发任务"30 10 1 20 * ? &quo…

【UCIe】UCIe D2D Adapter 介绍

&#x1f525;点击查看精选 UCIe 系列文章&#x1f525; &#x1f525;点击进入【芯片设计验证】社区&#xff0c;查看更多精彩内容&#x1f525; &#x1f4e2; 声明&#xff1a; &#x1f96d; 作者主页&#xff1a;【MangoPapa的CSDN主页】。⚠️ 本文首发于CSDN&#xff0…

springboot大学生兼职网站毕业设计源码311734

springboot大学生兼职网站 摘 要 信息化社会内需要与之针对性的信息获取途径&#xff0c;但是途径的扩展基本上为人们所努力的方向&#xff0c;由于站在的角度存在偏差&#xff0c;人们经常能够获得不同类型信息&#xff0c;这也是技术最为难以攻克的课题。针对大学生兼职网站等…

React + Dva + Antd + Umi 快速入门

最近一个项目用了React + Dva + Antd + Umi 技术栈基础框架概念 React前端三大框架之一。Dva由阿里架构师 sorrycc 带领 team 完成的一套前端框架,在作者的 github 里是这么描述它的:”dva 是 react 和 redux 的最佳实践”。Antd是阿里的一套开箱即用的中台前端/设计解决方案…

Vue模板语法上集(02)

今日份分享内容&#xff1a; 一、插值&#xff08;该代码块会放在末尾一并展示&#xff09; 1、文本插值 2、使用v-html指令用于输出html代码 3、属性 HTML属性中的值应使用v-bind指令 4、表达式 5、class 样式绑定 二、指令 1、 v-if 2、 v-show 3、v-for&…

Docker高级-1.复杂安装示例(mysql主从复制、redis集群)

目录 一、mysql主从复制 1.1 主服务器 1.2 从服务器 二、redis集群 2.1 问题引入-1~2亿条数据需要缓存&#xff0c;如何设计这个存储案例 2.1.1 方案一-哈希取余分区 2.1.2 方案二-一致性哈希算法分区 2.1.3 方案三-哈希槽分区 2.2 redis集群搭建演示 2.3 数据读写测试…

mybatis-plus-generator 配置不生成 entity, controller, mapper 等

3.5.2版本 有需求不生成controller 于是baidu 发现如下方法.templateConfig(builder -> builder.controller(""))配置后确实不生成controller又有需求不生成entity 尝试以下代码未果.templateConfig(builder -> builder.entity(""))于是查看源代码和…

【编程题】【Scratch二级】2022.06 画正方形

画正方形 在舞台正中央绘制一个边长为200的正方形。 1. 准备工作 &#xff08;1&#xff09;保留默认小猫角色并隐藏角色&#xff1b; &#xff08;2&#xff09;默认空白背景&#xff1b; &#xff08;3&#xff09;添加画笔模块。 2. 功能实现 &#xff08;1&#xff…

K8s(kubernetes)介绍以及原理解析

K8s&#xff08;kubernetes&#xff09; 云原生 服务部署模式 物理机模式–>虚拟化模式–>云端模式&#xff08;云原生模式&#xff09; K8s简介及架构 容器编排技术&#xff0c;用来管理容器 但是不直接管理容器&#xff0c;通过管理pod来间接管理容器 pod是k8s最小…

Android的handler消息收发处理——子线程与主线程(UI线程)间的通信

目录 写在前面 基础概念 什么是handler&#xff1f; 什么是looper&#xff1f; 什么是消息队列&#xff08;MessageQueue&#xff09;&#xff1f; 在子线程中使用子线程中的数据更新UI线程 主线程与子线程通信实例&#xff08;程序代码&#xff09; 子线程获取主线程h…