V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
hansonwang99
V2EX  ›  程序员

前后端分离实践:基于 vue 实现网站前台的权限管理

  •  
  •   hansonwang99 ·
    hansonwang99 · 2018-03-10 11:05:55 +08:00 · 3610 次点击
    这是一个创建于 2433 天前的主题,其中的信息可能已经有所发展或是发生改变。

    LG


    Javascript 做为当下的热门语言,用途很广泛,从前端到后端处处可见其存在,该技术如今在我们项目内部也大量使用来开发诸如 CMS 系统以及其他其他一些数据分析系统的前端页面,为此个人非常感兴趣并将其作为帽子卡的扩展内容来进行课余学习。


    Javascript 框架鳞次栉比,但基本原理大致相同,因此选用国内人开发的 vue.js 进行一个初步的尝试。学习 vue.js 也一周多的时间了,说起 vue 的主要用法,无外乎 Declarative Rendering、Component System、Client-side Routing、Vue-resource、Axios 以及视项目大小而决定是否使用的 Vuex,学习 vue 事小,主要转变思维,面向前后端分离的组件式 web 开发才是真正想去实践的。


    正好我的个人网站 CodeSheep 最近要开发后台管理,因此正好用 vue 这一套来实现了一下。说到后台管理,绕不开的问题就是权限的管理。既然想实践前后端分离这种思想,因此后台管理的所有 web 前端的东西应该独立由前端完成,这其中就包括很重要的由前端来根据权限进行相关东西的控制。我们想要做到的是:不同的权限对应着不同的路由,同时页面侧边栏也应该根据不同的权限,来异步生成对应的菜单,讲白了就是后台管理时不同权限的用户其看到的界面菜单是不一样的,因此有了这里实现登录和权限验证的一套流程。

    ##具体实现 ###1、点击“登录”按钮触发登录事件

    this.$store.dispatch('LoginByEmail', this.loginForm).then(() => {
      this.$router.push({ path: '/' }); //登录成功之后重定向到首页
    }).catch(err => {
      this.$message.error(err); //登录失败提示错误
    });
    

    其中异步触发的 actions LoginByEmail 的处理内容如下:

    LoginByEmail ({ commit }, userInfo) {
          const email = userInfo.email.trim()
          return new Promise((resolve, reject) => {
            loginByEmail(email, userInfo.password).then(response => {
              const data = response.data
              setToken(response.data.token)
              commit('SET_TOKEN', data.token)
              resolve()
            }).catch(error => {
              reject(error)
            })
          })
        }
    

    很容易看出想做的是将从服务器端拿到的 token (唯一标示用户身份)放到浏览器本地 Cookie 中去

    ###2、全局钩子 router.beforeEach 中拦截路由 这一步是核心,具体处理流程示意如下:

    路由拦截处理流程

    具体代码如下:

    router.beforeEach((to, from, next) => {
      if (getToken()) {  // 判断是否取到 token
        if (to.path === '/login') {
          next({ path: '/' })
        } else {
          if (store.getters.roles.length === 0) {  // 判断当前用户是否已获取完 user_info 信息
            store.dispatch('GetInfo').then(res => { // 获取 user_info
              const roles = res.data.role
              store.dispatch('GenerateRoutes', { roles }).then(() => { // 生成可访问的路由表
                router.addRoutes(store.getters.addRouters)  // 动态添加可访问路由表
                next({ ...to }) // 放行路由
              })
            }).catch(() => {
              store.dispatch('FedLogOut').then(() => {
                next({ path: '/login' })
              })
            })
          } else {
            next() // 放行该路由
          }
        }
      } else {
        if (whiteList.indexOf(to.path) !== -1) { // 在免登录白名单里的路径,继续让其访问
          next()
        } else { // 其他不在白名单里的路径全部让其重定向到登录页面!
          next('/login')
          alert('not in white list, now go to the login page')
        }
      }
    })
    

    流程图中几个重要步骤解释一下:

    • 判断前端是否取到了 token 令牌:getToken()

    操作很简单,主要是从 Cookie 中获取,看 token 是不是已经拿到了:

    export function getToken () {
      return Cookies.get(TokenKey)
    }
    
    • vuex 异步操作 store.dispatch('GetInfo'):获取用户信息
        GetInfo ({ commit, state }) {
          return new Promise((resolve, reject) => {
            getInfo(state.token).then(response => {
              const data = response.data
              console.log(data)
              commit('SET_ROLES', data.role)
              commit('SET_NAME', data.name)
              resolve(response)
            }).catch(error => {
              reject(error)
            })
          })
        }
    

    操作也很简单,用一个 get 的 restful api 从服务器获取用户的角色和名字

    • vuex 异步操作 store.dispatch('GenerateRoutes', { roles }):根据不同的 roles 来生成不同的前台路由
        GenerateRoutes ({ commit }, data) {
          return new Promise(resolve => {
            const { roles } = data
            let accessedRouters
            if (roles.indexOf('admin') >= 0) {
              accessedRouters = asyncRouter
            } else {
              accessedRouters = filterAsyncRouter(asyncRouter, roles)
            }
            commit('SET_ROUTERS', accessedRouters)
            resolve()
          })
        }
    

    从代码中可以看出,我这是只区分了管理员角色 admin 和其他普通用户(即非 Aadmin 两种权限)

    该系列的实践后续还将尝试更多,将会一一撰帖成文,我也是个初学者,路漫漫而求索之。。。


    后记

    作者更多的原创文章在此


    7 条回复    2018-03-13 00:49:17 +08:00
    leemove
        1
    leemove  
       2018-03-10 15:22:24 +08:00
    哇 我做过一个更复杂的....但是没系统整理过...
    notreami
        2
    notreami  
       2018-03-10 22:28:40 +08:00
    现在越来越反感前后端分离的口号,也越来越反感这种 面向 webpack 配置 编程。不是认为这样不好,而且人力不够的情况,没看到好处在哪?而且就前后端分离,后端起个只有模版引擎的项目也能达到这样的效果,为什么要用这么麻烦的配置呢?
    Linxing
        3
    Linxing  
       2018-03-11 00:24:35 +08:00 via iPhone
    学习了 感谢 一直想学 vue 跟我的 flask 搭配
    iyangyuan
        4
    iyangyuan  
       2018-03-11 14:05:27 +08:00 via iPhone
    前后端分离解决的不是技术问题,而且协作问题
    LeungJZ
        5
    LeungJZ  
       2018-03-12 14:11:09 +08:00
    hansonwang99
        6
    hansonwang99  
    OP
       2018-03-12 15:10:46 +08:00
    @LeungJZ 这代码就原作者会写?
    bingod
        7
    bingod  
       2018-03-13 00:49:17 +08:00
    很棒,感谢分享!
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   996 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 20ms · UTC 21:30 · PVG 05:30 · LAX 13:30 · JFK 16:30
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.