angular基于路由控制ui-router實(shí)現(xiàn)系統(tǒng)權(quán)限控制
前端去實(shí)現(xiàn)權(quán)限控制聽(tīng)起來(lái)有點(diǎn)扯淡(實(shí)際也有點(diǎn)扯淡),掩耳盜鈴,主要是擔(dān)心安全問(wèn)題,但是如果在前后端分離的情況下,需要做一個(gè)帶有權(quán)限控制的后臺(tái)管理系統(tǒng),angular基于ui-router應(yīng)該怎么做呢?
權(quán)限的設(shè)計(jì)中比較常見(jiàn)的就是RBAC基于角色的訪問(wèn)控制,基本思想是,對(duì)系統(tǒng)操作的各種權(quán)限不是直接授予具體的用戶,而是在用戶集合與權(quán)限集合之間建立一個(gè)角色集合。每一種角色對(duì)應(yīng)一組相應(yīng)的權(quán)限。
一旦用戶被分配了適當(dāng)?shù)慕巧螅撚脩艟蛽碛写私巧乃胁僮鳈?quán)限。這樣做的好處是,不必在每次創(chuàng)建用戶時(shí)都進(jìn)行分配權(quán)限的操作,只要分配用戶相應(yīng)的角色即可,而且角色的權(quán)限變更比用戶的權(quán)限變更要少得多,這樣將簡(jiǎn)化用戶的權(quán)限管理,減少系統(tǒng)的開(kāi)銷(xiāo)。
前端基于angular開(kāi)發(fā)的管理系統(tǒng)在權(quán)限控制方面需要處理如下幾種情況:
1、UI,該用戶對(duì)應(yīng)的角色權(quán)限能夠訪問(wèn)哪些頁(yè)面不能訪問(wèn)哪些頁(yè)面;
2、理由,當(dāng)用戶準(zhǔn)備跳轉(zhuǎn)到某一頁(yè)面時(shí),如果沒(méi)有該頁(yè)面的訪問(wèn)權(quán),重定向到一個(gè)錯(cuò)誤提示頁(yè)面;
3、http請(qǐng)求,該用戶是否具有訪問(wèn)某些API的權(quán)限,如果沒(méi)有返回403
如果處理以上問(wèn)題?
大致思路是:
1、用戶登錄后從server獲取該用戶的permission
2、然后建立一個(gè)service來(lái)保存該permission的映射關(guān)系,比如a用戶的permission是查看page1,page2,那么在路由發(fā)生變更之前判斷是否有訪問(wèn)的權(quán)限,有正常跳轉(zhuǎn),沒(méi)有則跳轉(zhuǎn)到統(tǒng)一的403頁(yè)面或者其他。
3、對(duì)于http請(qǐng)求,可以讓server來(lái)處理,判斷用戶是否有請(qǐng)求權(quán)限
獲取用戶permission,比如:
var permissionList; angular.element(document).ready(function() { $.get('/api/UserPermission', function(data) { permissionList = data; angular.bootstrap(document, ['App']); }); });
這里用到的是jquery的ajax,因?yàn)樵诖酥癮ngular還沒(méi)啟動(dòng),如果我們的登錄也是用angular實(shí)現(xiàn),可以在登錄之后讓server
返回permission
,然后保存下來(lái)。
判斷該用戶是否具有某個(gè)權(quán)限,比如:
app.factory('permissions', function($rootScope) { return { hasPermission: function(permission) { if (permission) { if (typeof(permission) == "string") { if (permissionList.indexOf(permission) > -1) { return true; } } } return false; } }; });
路由權(quán)限控制:
app.run(function($rootScope, $location,$state, permissions) { $rootScope.$on('$stateChangeStart', function(event, toState, toParams, fromState, fromParams) { var permission = toState.permission; if (toState.name!="login"&&!permissions.hasPermission(permission)) { // event.preventDefault(); // $state.transitionTo("login"); } }); });
//路由配置 app.config(function($stateProvider, $urlRouterProvider, $controllerProvider) { app.registerController = $controllerProvider.register; app.loadJs = function(js) { return function($rootScope, $q) { var def = $q.defer(), deps = []; angular.isArray(js) ? (deps = js) : deps.push(js); require(deps, function() { $rootScope.$apply(function() { def.resolve(); }); }); return def.promise; }; }; $urlRouterProvider.otherwise('/login'); $stateProvider.state('login', { url: '/login', templateUrl: '/assets/console/pages/login.html', controller: 'loginController', resolve: { deps: app.loadJs('./controllers/login') } }); $stateProvider.state('index', { url: '/index', templateUrl: '/assets/console/pages/home.html', controller: 'indexController', resolve: { deps: app.loadJs('./controllers/index') }, permission: "super_admin" }); });
開(kāi)發(fā)過(guò)程中實(shí)際會(huì)遇到的問(wèn)題:
1、登錄后如何刷新頁(yè)面,因?yàn)槲覀兊牡卿浶畔⒉糠质莝erver框架實(shí)現(xiàn)的,沒(méi)完全分離,所以登錄后登錄信息沒(méi)有刷新,可以通過(guò)判斷fromState和toState來(lái)判斷是否從登錄頁(yè)面跳轉(zhuǎn)到指定頁(yè)面,然后通過(guò) $window.location.reload();
實(shí)現(xiàn)頁(yè)面的整體刷新。
2、跳轉(zhuǎn)后當(dāng)前導(dǎo)航的選中狀態(tài)更新,state成功后刷新UI
app.run(['$rootScope', "$state", '$window', '$location', '$log', function($rootScope, $state, $window, $location, $log) { $rootScope.$on('$stateChangeSuccess', function(evt, toState, roParams, fromState, fromParams) { //如果是登錄進(jìn)來(lái)就刷新頁(yè)面 setTimeout(function(){ appCommon.initUI(); },500); }); }]);
相關(guān)文章
AngularJs Injecting Services Into Controllers詳解
本文主要介紹AngularJs Injecting Services Into Controllers的知識(shí),這里整理了一下相關(guān)資料,及示例代碼,幫助大家學(xué)習(xí)和理解,有興趣的小伙伴可以參考下2016-09-09詳解AngularJs ui-router 路由的簡(jiǎn)單介紹
本篇文章主要介紹了AngularJs ui-router 路由的簡(jiǎn)單介紹。簡(jiǎn)單明了的說(shuō)明了ngRoute和ui-router的區(qū)別和用法,有興趣的可以了解一下2017-04-04詳解創(chuàng)建自定義的Angular Schematics
本文對(duì) Angular Schematics 進(jìn)行了介紹,并創(chuàng)建了一個(gè)用于創(chuàng)建自定義 Component 的 Schematics ,然后在 Angular 項(xiàng)目中以它為模板演練了通過(guò) Schematics 添加自定義的 Component,感興趣的小伙伴們可以參考一下2018-06-06AngularJS基礎(chǔ) ng-mouseover 指令簡(jiǎn)單示例
本文主要介紹AngularJS ng-mouseover 指令,這里幫大家整理了AngularJS 指令的基礎(chǔ)知識(shí),并附代碼示例,有興趣的小伙伴可以參考下2016-08-08Angular2學(xué)習(xí)教程之TemplateRef和ViewContainerRef詳解
這篇文章主要給大家介紹了Angular2中TemplateRef和ViewContainerRef的相關(guān)資料,文中介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-05-05angularjs項(xiàng)目的頁(yè)面跳轉(zhuǎn)如何實(shí)現(xiàn)(5種方法)
本篇文章主要介紹了詳解angularjs項(xiàng)目的頁(yè)面跳轉(zhuǎn)如何實(shí)現(xiàn) ,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05詳解Angular調(diào)試技巧之報(bào)錯(cuò)404(not found)
本篇文章主要介紹了詳解Angular調(diào)試技巧之報(bào)錯(cuò)404(not found),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Angular directive遞歸實(shí)現(xiàn)目錄樹(shù)結(jié)構(gòu)代碼實(shí)例
本篇文章主要介紹了Angular directive遞歸實(shí)現(xiàn)目錄樹(shù)結(jié)構(gòu)代碼實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05Angular實(shí)踐之將Input與Lifecycle轉(zhuǎn)換成流示例詳解
這篇文章主要為大家介紹了Angular實(shí)踐之將Input與Lifecycle轉(zhuǎn)換成流示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02