前后端分离实践:基于 vue 实现网站前台的权限管理 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
请不要在回答技术问题时复制粘贴 AI 生成的内容
hansonwang99
V2EX    程序员

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

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

    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     Solana     872 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 28ms UTC 20:59 PVG 04:59 LAX 13:59 JFK 16:59
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86