首页 教程 Web前端 Vue3+Vite怎么使用双token实现无感刷新

Vue3+Vite怎么使用双token实现无感刷新

一、token 登录鉴权

jwt:JSON Web Token。是一种认证协议,一般用来校验请求的身份信息和身份权限。 由三部分组成:Header、Hayload、Signature

header:也就是头部信息,是描述这个 token 的基本信息,json 格式

{   "alg": "HS256", // 表示签名的算法,默认是 HMAC SHA256(写成 HS256)   "type": "JWT" // 表示Token的类型,JWT 令牌统一写为JWT }

payload:载荷,也是一个 JSON 对象,用来存放实际需要传递的数据。不建议存放敏感信息,比如密码。

{   "iss": "a.com", // 签发人   "exp": "1d", // expiration time 过期时间   "sub": "test", // 主题   "aud": "", // 受众   "nbf": "", // Not Before 生效时间   "iat": "", // Issued At 签发时间   "jti": "", // JWT ID 编号   // 可以定义私有字段   "name": "",   "admin": "" }

Signature 签名 是对前两部分的签名,防止数据被篡改。 需要指定一个密钥。这个密钥只有服务器才知道,不能泄露。使用 Header 里面指定的签名算法,按照公式产生签名。

算出签名后,把 Header、Payload、Signature 三个部分拼成的一个字符串,每个部分之间用 . 分隔。这样就生成了一个 token

二、何为双 token

accessToken:用户获取数据权限

refreshToken:用来获取新的accessToken

双 token 验证机制,其中 accessToken 过期时间较短,refreshToken 过期时间较长。当 accessToken 过期后,使用 refreshToken 去请求新的 token。

双 token 验证流程

用户登录向服务端发送账号密码,登录失败返回客户端重新登录。登录成功服务端生成 accessToken 和 refreshToken,返回生成的 token 给客户端。

在请求拦截器中,请求头中携带 accessToken 请求数据,服务端验证 accessToken 是否过期。token 有效继续请求数据,token 失效返回失效信息到客户端。

客户端收到服务端发送的请求信息,在二次封装的 axios 的响应拦截器中判断是否有 accessToken 失效的信息,没有返回响应的数据。有失效的信息,就携带 refreshToken 请求新的 accessToken。

服务端验证 refreshToken 是否有效。有效,重新生成 token, 返回新的 token 和提示信息到客户端,无效,返回无效信息给客户端。

客户端响应拦截器判断响应信息是否有 refreshToken 有效无效。无效,退出当前登录。有效,重新存储新的 token,继续请求上一次请求的数据。

注意事项

短token失效,服务端拒绝请求,返回token失效信息,前端请求到新的短token如何再次请求数据,达到无感刷新的效果。

服务端白名单,成功登录前是还没有请求到token的,那么如果服务端拦截请求,就无法登录。定制白名单,让登录无需进行token验证。

三、服务端代码

1、搭建koa2服务器

全局安装koa脚手架

npm install koa-generator -g

创建服务端 直接koa2+项目名

koa2 server

cd server 进入到项目安装jwt

npm i jsonwebtoken

为了方便直接在服务端使用koa-cors 跨域

npm i koa-cors

在app.js中引入应用cors

const cors=require('koa-cors') ... app.use(cors())

2、双token

新建utils/token.js

const jwt=require('jsonwebtoken')   const secret='2023F_Ycb/wp_sd'  // 密钥 /* expiresIn:5 过期时间,时间单位是秒 也可以这么写 expiresIn:1d 代表一天  1h 代表一小时 */ // 本次是为了测试,所以设置时间 短token5秒 长token15秒 const accessTokenTime=5   const refreshTokenTime=15    // 生成accessToken const setAccessToken=(payload={})=>{  // payload 携带用户信息     return jwt.sign(payload,secret,{expireIn:accessTokenTime}) } //生成refreshToken const setRefreshToken=(payload={})=>{     return jwt.sign(payload,secret,{expireIn:refreshTokenTime}) }   module.exports={     secret,     setAccessToken,     setRefreshToken }

3、路由

直接使用脚手架创建的项目已经在app.js使用了路由中间件 在router/index.js 创建接口

const router = require('koa-router')() const jwt = require('jsonwebtoken') const { getAccesstoken, getRefreshtoken, secret }=require('../utils/token')   /*登录接口*/ router.get('/login',()=>{     let code,msg,data=null     code=2000     msg='登录成功,获取到token'     data={         accessToken:getAccessToken(),         refreshToken:getReferToken()     }     ctx.body={         code,         msg,         data     } })   /*用于测试的获取数据接口*/ router.get('/getTestData',(ctx)=>{     let code,msg,data=null     code=2000     msg='获取数据成功'     ctx.body={         code,         msg,         data     } })   /*验证长token是否有效,刷新短token   这里要注意,在刷新短token的时候回也返回新的长token,延续长token,   这样活跃用户在持续操作过程中不会被迫退出登录。长时间无操作的非活   跃用户长token过期重新登录 */ router.get('/refresh',(ctx)=>{     let code,msg,data=null     //获取请求头中携带的长token     let r_tk=ctx.request.headers['pass']     //解析token 参数 token 密钥 回调函数返回信息     jwt.verify(r_tk,secret,(error)=>{         if(error){             code=4006,             msg='长token无效,请重新登录'         } else{             code=2000,             msg='长token有效,返回新的token',             data={                 accessToken:getAccessToken(),                 refreshToken:getReferToken()             }         }     }) })

4、应用中间件

utils/auth.js

const { secret } = require('./token') const jwt = require('jsonwebtoken')   /*白名单,登录、刷新短token不受限制,也就不用token验证*/ const whiteList=['/login','/refresh'] const isWhiteList=(url,whiteList)=>{         return whiteList.find(item => item === url) ? true : false }   /*中间件  验证短token是否有效 */ const cuth = async (ctx,next)=>{     let code, msg, data = null     let url = ctx.path     if(isWhiteList(url,whiteList)){         // 执行下一步         return await next()     } else {         // 获取请求头携带的短token         const a_tk=ctx.request.headers['authorization']         if(!a_tk){             code=4003             msg='accessToken无效,无权限'             ctx.body={                 code,                 msg,                 data             }         } else{             // 解析token             await jwt.verify(a_tk,secret.(error)=>{                 if(error)=>{                       code=4003                       msg='accessToken无效,无权限'                       ctx.body={                           code,                           msg,                           datta                       }                 } else {                     // token有效                     return await next()                 }             })         }     } } module.exports=auth

在app.js中引入应用中间件

const auth=requier(./utils/auth) ··· app.use(auth)

其实如果只是做一个简单的双token验证,很多中间件是没必要的,比如解析静态资源。不过为了节省时间,方便就直接使用了koa2脚手架。

最终目录结构:

Vue3+Vite怎么使用双token实现无感刷新

四、前端代码

1、Vue3+Vite框架

前端使用了Vue3+Vite的框架,看个人使用习惯。

npm init vite@latest client_side

安装axios

npm i axios2、定义使用到的常量

config/constants.js

export const ACCESS_TOKEN = 'a_tk' // 短token字段 export const REFRESH_TOKEN = 'r_tk' // 短token字段 export const AUTH = 'Authorization'  // header头部 携带短token export const PASS = 'pass' // header头部 携带长token

3、存储、调用过期请求

关键点:把携带过期token的请求,利用Promise存在数组中,保持pending状态,也就是不调用resolve()。当获取到新的token,再重新请求。 utils/refresh.js

export {REFRESH_TOKEN,PASS} from '../config/constants.js' import { getRefreshToken, removeRefreshToken, setAccessToken, setRefreshToken} from '../config/storage'   let subsequent=[] let flag=false // 设置开关,保证一次只能请求一次短token,防止客户多此操作,多次请求   /*把过期请求添加在数组中*/ export const addRequest = (request) => {     subscribes.push(request) }   /*调用过期请求*/ export const retryRequest = () => {     console.log('重新请求上次中断的数据');     subscribes.forEach(request => request())     subscribes = [] }   /*短token过期,携带token去重新请求token*/ export const refreshToken=()=>{     if(!flag){         flag = true;         let r_tk = getRefershToken() // 获取长token         if(r_tk){             server.get('/refresh',Object.assign({},{                 headers:{[PASS]=r_tk}             })).then((res)=>{                 //长token失效,退出登录                 if(res.code===4006){                     flag = false                     removeRefershToken(REFRESH_TOKEN)                 } else if(res.code===2000){                     // 存储新的token                     setAccessToken(res.data.accessToken)                     setRefreshToken(res.data.refreshToken)                     flag = false                     // 重新请求数据                     retryRequest()                 }             })         }     } }4、封装axios

utlis/server.js

import axios from "axios"; import * as storage from "../config/storage" import * as constants from '../config/constants' import { addRequest, refreshToken } from "./refresh";   const server = axios.create({     baseURL: 'http://localhost:3004', // 你的服务器     timeout: 1000 * 10,     headers: {         "Content-type": "application/json"     } })   /*请求拦截器*/ server.interceptors.request.use(config => {     // 获取短token,携带到请求头,服务端校验     let aToken = storage.getAccessToken(constants.ACCESS_TOKEN)     config.headers[constants.AUTH] = aToken     return config })   /*响应拦截器*/ server.interceptors.response.use(     async response => {         // 获取到配置和后端响应的数据         let { config, data } = response         console.log('响应提示信息:', data.msg);         return new Promise((resolve, reject) => {             // 短token失效             if (data.code === 4003) {                 // 移除失效的短token                 storage.removeAccessToken(constants.ACCESS_TOKEN)                 // 把过期请求存储起来,用于请求到新的短token,再次请求,达到无感刷新                 addRequest(() => resolve(server(config)))                 // 携带长token去请求新的token                 refreshToken()             } else {                 // 有效返回相应的数据                 resolve(data)             }           })       },     error => {         return Promise.reject(error)     } )

5、复用封装

import * as constants from "./constants"   // 存储短token export const setAccessToken = (token) => localStorage.setItem(constanst.ACCESS_TOKEN, token) // 存储长token export const setRefershToken = (token) => localStorage.setItem(constants.REFRESH_TOKEN, token) // 获取短token export const getAccessToken = () => localStorage.getItem(constants.ACCESS_TOKEN) // 获取长token export const getRefershToken = () => localStorage.getItem(constants.REFRESH_TOKEN) // 删除短token export const removeAccessToken = () => localStorage.removeItem(constants.ACCESS_TOKEN) // 删除长token export const removeRefershToken = () => localStorage.removeItem(constants.REFRESH_TOKEN)

6、接口封装

apis/index.js

import server from "../utils/server"; /*登录*/ export const login = () => {     return server({         url: '/login',         method: 'get'     }) } /*请求数据*/ export const getData = () => {     return server({         url: '/getList',         method: 'get'     }) }

项目运行

Vue3+Vite怎么使用双token实现无感刷新

最后的最后,运行项目,查看效果 后端设置的短token5秒,长token10秒。登录请求到token后,请求数据可以正常请求,五秒后再次请求,短token失效,这时长token有效,请求到新的token,refresh接口只调用了一次。长token也过期后,就需要重新登录啦。

Vue3+Vite怎么使用双token实现无感刷新

评论(0)条

提示:请勿发布广告垃圾评论,否则封号处理!!

    猜你喜欢
    【MySQL】用户管理

    【MySQL】用户管理

     服务器/数据库  2个月前  2.15k

    我们推荐使用普通用户对数据的访问。而root作为管理员可以对普通用户对应的权限进行设置和管理。如给张三和李四这样的普通用户权限设定后。就只能操作给你权限的库了。

    Cursor Rules 让开发效率变成10倍速

    Cursor Rules 让开发效率变成10倍速

     服务器/数据库  2个月前  1.21k

    在AI与编程的交汇点上,awesome-cursorrules项目犹如一座灯塔,指引着开发者们驶向更高效、更智能的编程未来。无论你是经验丰富的老手,还是刚入行的新人,这个项目都能为你的编程之旅增添一抹亮色。这些规则文件就像是你私人定制的AI助手,能够根据你的项目需求和个人偏好,精确地调教AI的行为。突然间,你会发现AI不仅能理解Next.js的最佳实践,还能自动应用TypeScript的类型检查,甚至主动提供Tailwind CSS的类名建议。探索新的应用场景,推动AI辅助编程的边界。

    探索Django 5: 从零开始,打造你的第一个Web应用

    探索Django 5: 从零开始,打造你的第一个Web应用

     服务器/数据库  2个月前  1.12k

    Django 是一个开放源代码的 Web 应用程序框架,由 Python 写成。它遵循 MVT(Model-View-Template)的设计模式,旨在帮助开发者高效地构建复杂且功能丰富的 Web 应用程序。随着每个版本的升级,Django 不断演变,提供更多功能和改进,让开发变得更加便捷。《Django 5 Web应用开发实战》集Django架站基础、项目实践、开发经验于一体,是一本从零基础到精通Django Web企业级开发技术的实战指南《Django 5 Web应用开发实战》内容以。

    MySQL 的mysql_secure_installation安全脚本执行过程介绍

    MySQL 的mysql_secure_installation安全脚本执行过程介绍

     服务器/数据库  2个月前  1.08k

    mysql_secure_installation 是 MySQL 提供的一个安全脚本,用于提高数据库服务器的安全性

    【MySQL基础篇】概述及SQL指令:DDL及DML

    【MySQL基础篇】概述及SQL指令:DDL及DML

     服务器/数据库  2个月前  482

    数据库是长期存储在计算机内的、有组织的、可共享的、统一管理的大量数据的集合。数据库不仅仅是数据的简单堆积,而是遵循一定的规则和模式进行组织和管理的。数据库中的数据可以包括文本、数字、图像、音频等各种类型的信息。

    Redis中的哨兵(Sentinel)

    Redis中的哨兵(Sentinel)

     服务器/数据库  2个月前  308

    ​ 上篇文章我们讲述了Redis中的主从复制(Redis分布式系统中的主从复制-CSDN博客),本篇文章针对主从复制中的问题引出Redis中的哨兵,希望本篇文章会对你有所帮助。