热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

vueelementadmin登出切换用户后重新登录跳转404页面Bug解决记录

01Bug描述笔者基于简化版的vue-element-admin前端框架vue-admin-template进行二次开发。我在项目中设定了三个用户角色,不同的角色具有不同的权限,在

01 Bug 描述

笔者基于简化版的 vue-element-admin 前端框架 vue-admin-template 进行二次开发。

我在项目中设定了三个用户角色,不同的角色具有不同的权限,在此之前项目中已经实现了不同用户角色的权限认证以及动态路由生成:vue-element-admin 动态路由无法动态渲染侧边栏-解决记录。

加上权限认证之后,项目就出现的了一个十分魔幻的 Bug:我使用用户A登录系统后,在系统内选择退出登录,回到登录页面后切换用户B登录系统,这时直接跳转到404页面。如下图所示:

在这里插入图片描述

可以看到1 中我已经成功登录了管理员用户,并在2中显示了我已经成功退出登录了管理员用户,然后我在3中切换为学生用户登录系统。

点击登录后,出现如下图所示情况,可以看到学生用户已经成功登录,但是页面却没有成功跳转,而是在404页面。更加神奇的是,这个Bug 时有时无,并不是每次切换用户都会出现这种情况,让人摸不着头脑。

在这里插入图片描述

02 登录权限流程

毫无疑问,登出切换用户后重新登录跳转404页面的Bug肯定是出现在登录逻辑权限认证过程中,所以接下来我们梳理一下 vue-element-admin 登录逻辑和权限认证流程。

2.1 vue-element-admin 登录逻辑

vue-element-admin 登录逻辑如下图所示:

  • 首先在登录页面点击按钮后触发点击事件,然后在事件中分发store.action
  • 在 store 的对应 action 中调用 axios 接口获取后台数据,如果登录成功则更具用户角色创建 token,并将这些认证信息保存到 COOKIEs 中
  • 获取 token 之后在@/src/permission.js 获取用户信息并进行权限认证生成动态路由

在这里插入图片描述

2.1.1 登录页面点击按钮后触发点击事件

@/src/views/login/index.vue 中查看登录页面点击按钮后触发点击事件,并在事件中分发 store.action 实现相关代码如下:

<template><divclass="login-container"><el-form ref="loginForm":model="loginForm":rules="loginRules"class="login-form" autocomplete="on" label-position="left"><--! 登录页面点击按钮后触发点击事件--><el-button:loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">Login</el-button></el-form></div></template><script>import{ validUsername}from'@/utils/validate'exportdefault{
  name:'Login',
  watch:{
    $route:{handler:function(route){this.redirect= route.query&& route.query.redirect},
      immediate:true}},
  methods:{// 在事件中分发 store.actionhandleLogin(){this.$refs.loginForm.validate(valid=>{if(valid){this.loading=true// 分发 store.actionthis.$store.dispatch('user/login',this.loginForm).then(()=>{this.$router.push({ path:this.redirect||'/'})this.loading=false}).catch(()=>{this.loading=false})}else{
          console.log('error submit!!')returnfalse}})}}}</script>

2.1.2 在 store 的对应 action 中调用 axios 接口获取后台数据

@/src/store/modules/user.js 的对应 action 中调用 axios 接口获取后台数据,相关实现代码如下:

import{ login, logout, getInfo}from'@/api/user'import{ getToken, setToken, removeToken}from'@/utils/auth'import router,{ resetRouter}from'@/router'const state={
  token:getToken(),
  name:'',
  avatar:'',
  introduction:'',
  roles:[]}const mutations={SET_TOKEN:(state, token)=>{
    state.token= token},SET_INTRODUCTION:(state, introduction)=>{
    state.introduction= introduction},SET_NAME:(state, name)=>{
    state.name= name},SET_AVATAR:(state, avatar)=>{
    state.avatar= avatar},SET_ROLES:(state, roles)=>{
    state.roles= roles}}const actions={// 登录 actionlogin({ commit}, userInfo){const{ username, password}= userInforeturnnewPromise((resolve, reject)=>{// 在登录 action 中调用 axios 接口获取后台数据,如果获取成功则创建 tocken 并保存在 COOKIE 中login({ username: username.trim(), password: password}).then(response=>{const{ data}= responsecommit('SET_TOKEN', data.token)setToken(data.token)resolve()}).catch(error=>{reject(error)})})},// 获取用户信息 actiongetInfo({ commit, state}){returnnewPromise((resolve, reject)=>{getInfo(state.token).then(response=>{const{ data}= responseif(!data){reject('Verification failed, please Login again.')}const{ roles, name, avatar, introduction}= dataif(!roles|| roles.length<=0){reject('getInfo: roles must be a non-null array!')}commit('SET_ROLES', roles)commit('SET_NAME', name)commit('SET_AVATAR', avatar)commit('SET_INTRODUCTION', introduction)resolve(data)}).catch(error=>{reject(error)})})}}exportdefault{
  namespaced:true,
  state,
  mutations,
  actions}

2.1.3 登录 axios 接口定义

@/src/api/user.js 查看登录的 axios 请求接口,实现代码如下:

import requestfrom'@/utils/request'// 登录 api 请求exportfunctionlogin(data){returnrequest({
    url:'/vue-element-admin/user/login',
    method:'post',
    data})}// 获取用户信息 api 请求exportfunctiongetInfo(token){returnrequest({
    url:'/vue-element-admin/user/info',
    method:'get',
    params:{ token}})}

2.1.4 获取 token 后获取用户信息

用户登录成功之后,在全局钩子router.beforeEach 中拦截路由,判断是否已获得token,在获得token 之后就在@/src/permission.js 获取用户信息,相关代码如下:

import routerfrom'./router'import storefrom'./store'import{ getToken}from'@/utils/auth'// get token from COOKIE

router.beforeEach(async(to,from, next)=>{const hasToken=getToken()if(hasToken){const hasRoles= store.getters.roles&& store.getters.roles.length>0if(hasRoles){next()}else{try{// 分发 store.action 获取用户信息const{ roles}=await store.dispatch('user/getInfo')const accessRoutes=await store.dispatch('permission/generateRoutes', roles)
          router.addRoutes(accessRoutes)next({...to, replace:true})}catch(error){await store.dispatch('user/resetToken')
          Message.error(error||'Has Error')next(`/login?redirect=${to.path}`)}}}})

2.2 vue-element-admin 权限认证流程

vue-element-admin 的权限认证流程入下图所示:

  • 在项目入口的@/src/main.js 中创建 vue 实例时,将 vue-router 挂载,但这个时候 vue-router 挂载的是全局路由表,即一些登录或者不用权限的公用的页面。
  • 当用户登录后验证是否携带 token,在@/src/permission.js 中分发user/getInfo 行为获取用户 role,然后将 role 作为输入分发permission/generateRoutes 行为将 role 和路由表每个页面的需要的权限作比较,生成最终用户可访问的动态路由表。
  • 调用router.addRoutes(store.getters.addRouters) 添加用户可访问的路由。
  • 使用 vuex 管理路由表,根据 vuex 中可访问的路由渲染侧边栏组件。

在这里插入图片描述

2.2.1 入口挂载全局路由表

在项目入口的@/src/main.js 中创建 vue 实例时,将 vue-router 挂载,但这个时候 vue-router 挂载的是全局路由表,即一些登录或者不用权限的公用的页面。相关代码如下:

import Vuefrom'vue'import Appfrom'./App'import storefrom'./store'// 引入全局路由表import routerfrom'./router'newVue({
  el:'#app',
  router,// 挂载全局路由表
  store,render:h=>h(App)})

2.2.2 权限认证

@/src/permission.js 中要完成如下权限认证任务:

  • 当用户登录后验证是否携带 token
  • 若携带则分发user/getInfo 行为获取用户 role 等相关信息
  • 然后将 role 作为输入分发permission/generateRoutes 行为,获取用户角色对应的动态路由表
  • 调用router.addRoutes(store.getters.addRouters) 添加用户可访问的路由

相关实现代码如下:

import routerfrom'./router'import storefrom'./store'import{ getToken}from'@/utils/auth'// get token from COOKIE

router.beforeEach(async(to,from, next)=>{// 验证是否携带 tokenconst hasToken=getToken()if(hasToken){// 已经登录 从 state 中获取用户角色const hasRoles= store.getters.roles&& store.getters.roles.length>0if(hasRoles){next()}else{try{// 首次登录,分发 user/getInfo 行为获取用户 role 等相关信息const{ roles}=await store.dispatch('user/getInfo')// 将 role 作为输入分发 `permission/generateRoutes` 行为,获取用户角色对应的动态路由表const accessRoutes=await store.dispatch('permission/generateRoutes', roles)// 添加用户可访问的路由
          router.addRoutes(accessRoutes)next({...to, replace:true})}catch(error){await store.dispatch('user/resetToken')
          Message.error(error||'Has Error')next(`/login?redirect=${to.path}`)}}}})

2.2.3 权限认证与动态路由实现

实现权限认证与动态路由大致过程分为如下四个步骤:

  • 修改src/store/modules/user.js 增加用户信息中角色的权限列表 roles

  • 修改src/router/index.js 根据用户角色划分路由

  • 增加src/store/modules/permission.js 通过获取当前用户的权限去比对路由表,生成当前用户具有访问权限的动态路由表

  • 修改src/permission.js 通过router.addRoutes 将用户可访问路由表动态挂载到 router 上

详细过程笔者已在vue-element-admin 动态路由无法动态渲染侧边栏-解决记录 中的第二节(动态路由修改过程) 介绍过了,有兴趣可以进一步了解。

03 发现 Bug

啰嗦啰嗦一大堆回顾了 vue-element-admin 登录权限的实现逻辑与代码,似乎并没有找到这个时有时无的魔幻 Bug 的出处,但是也找到了一些蛛丝马迹:

  • 出现在登录业务,且时有时无的 Bug 症状,说明一定与动态路由有关,只有动态变化才会出现一会儿正常一会儿魔怔
  • 能够成功登录,且返回了已经登录的用户信息,说明@/src/store/modules/user.js 中的用户登录行为以及用户信息获取行为都正确执行了;同时也说明@/src/permission.js 中的权限认证流程都成功执行了

那么 Bug 的原因就直指登录成功之后的页面跳转过程,登录成功后的跳转仅在登录页面点击按钮后触发点击事件 中有定义,即@/src/views/login/index.vue 中如下所示:

<template><divclass="login-container"><el-form ref="loginForm":model="loginForm":rules="loginRules"class="login-form" autocomplete="on" label-position="left"><--! 登录页面点击按钮后触发点击事件--><el-button:loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">Login</el-button></el-form></div></template><script>import{ validUsername}from'@/utils/validate'exportdefault{
  name:'Login',
  watch:{
    $route:{handler:function(route){this.redirect= route.query&& route.query.redirect},
      immediate:true}},
  methods:{// 在事件中分发 store.actionhandleLogin(){this.$refs.loginForm.validate(valid=>{if(valid){this.loading=true// 分发 store.actionthis.$store.dispatch('user/login',this.loginForm).then(()=>{// 登录成功后,跳转到 this.redirect 或者 / 路径下this.$router.push({ path:this.redirect||'/'})this.loading=false}).catch(()=>{this.loading=false})}else{
          console.log('error submit!!')returnfalse}})}}}</script>

Bug 的关键就在于这行代码this.$router.push({ path: this.redirect || '/' }) ,登录成功之后可以跳转到this.redirect

再看与其相关的watch 属性,一直在监听路由传的值和重定向路径,是不是已经发现了 Bug 所在?

watch:{
    $route:{handler:function(route){this.redirect= route.query&& route.query.redirect},
      immediate:true}}

04 解决 Bug

Bug 的成因

我通过如下示例来解释 Bug 的成因:

我们假设项目中全局路由即所有用户共用的路由是‘/’,'/login','/404'

用户 A 具有权限的路由是‘/userinfoA’,用户 B 具有权限的路由是‘/userinfoB

如果用户 A 登录之后并在路由‘/userinfoA’ 所指向的页面中登出跳转到登录页面,这时$route.query.redirect 保存的路由就是‘/userinfoA’

当在登出后的登录页面登录用户 B 时,此时的登录页面点击按钮后触发点击事件完成用户登录之后,在选择跳转页面时选择了路由‘/userinfoA’ 指向的页面,而用户 B 不具有该页面权限所以直接跳转到了 404 页面。

解决 Bug

一种简单的解决方式就是不使用登出前的路由,不管是首次登录还是切换用户登录,所有登录业务完成之后都跳转到共用路径‘/’ 所指向的页面,删除@/src/views/login/index.vue 中重定向相关即可,如下所示:

<template><divclass="login-container"><el-form ref="loginForm":model="loginForm":rules="loginRules"class="login-form" autocomplete="on" label-position="left"><--! 登录页面点击按钮后触发点击事件--><el-button:loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">Login</el-button></el-form></div></template><script>import{ validUsername}from'@/utils/validate'exportdefault{
  name:'Login',// watch: {//   $route: {//     handler: function(route) {//       this.redirect = route.query && route.query.redirect//     },//     immediate: true//   }// },
  methods:{// 在事件中分发 store.actionhandleLogin(){this.$refs.loginForm.validate(valid=>{if(valid){this.loading=true// 分发 store.actionthis.$store.dispatch('user/login',this.loginForm).then(()=>{// 登录成功后,跳转到 this.redirect 或者 / 路径下// this.$router.push({ path: this.redirect || '/' })this.$router.push({ path:'/'})this.loading=false}).catch(()=>{this.loading=false})}else{
          console.log('error submit!!')returnfalse}})}}}</script>

当然,你也可以通过在@/src/router/index.js 实现不保存redirect 的路由构建方法,可能会稍微复杂一些。

参考资料

vue-element-admin 官方文档:权限验证

手摸手,带你用vue撸后台 系列二(登录权限篇)

vue Element Admin 登录、验证流程


推荐阅读
  • 一、路由首先需要配置路由,就是点击good组件进入goodDetail组件配置路由如下{path:goodDetail,component:goodDetail}同时在good组件中写入如下点击事件,路由中加入 ... [详细]
  • 网络请求模块选择——axios框架的基本使用和封装
    本文介绍了选择网络请求模块axios的原因,以及axios框架的基本使用和封装方法。包括发送并发请求的演示,全局配置的设置,创建axios实例的方法,拦截器的使用,以及如何封装和请求响应劫持等内容。 ... [详细]
  • vue使用
    关键词: ... [详细]
  • 本文讨论了将HashRouter改为Router后,页面全部变为空白页且没有报错的问题。作者提到了在实际部署中需要在服务端进行配置以避免刷新404的问题,并分享了route/index.js中hash模式的配置。文章还提到了在vueJs项目中遇到过类似的问题。 ... [详细]
  • Vue基础一、什么是Vue1.1概念Vue(读音vjuː,类似于view)是一套用于构建用户界面的渐进式JavaScript框架,与其它大型框架不 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文介绍了如何按需加载elementui的部分模块,以及如何设置覆盖某些属性。通过import引入Dialog模块,并使用Vue.component进行全局设置。同时使用Vue.use引入ElementUI和VueAxios模块。通过extends进行属性覆盖设置。 ... [详细]
  • 本文介绍了如何使用vue-awesome-swiper组件,包括在main.js中引入和使用swiper和swiperSlide组件,以及设置options和ref属性。同时还介绍了如何在模板中使用swiper和swiperSlide组件,并展示了如何通过循环渲染swipes数组中的数据,并使用picUrl属性显示图片。最后还介绍了如何添加分页器。 ... [详细]
  • 本文详细介绍了cisco路由器IOS损坏时的恢复方法,包括进入ROMMON模式、设置IP地址、子网掩码、默认网关以及使用TFTP服务器传输IOS文件的步骤。 ... [详细]
  • 用Vue实现的Demo商品管理效果图及实现代码
    本文介绍了一个使用Vue实现的Demo商品管理的效果图及实现代码。 ... [详细]
  • 生成式对抗网络模型综述摘要生成式对抗网络模型(GAN)是基于深度学习的一种强大的生成模型,可以应用于计算机视觉、自然语言处理、半监督学习等重要领域。生成式对抗网络 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
author-avatar
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有