ASP.NET MVC許可權控制思路

来源:https://www.cnblogs.com/EvanYu/archive/2019/03/19/10556171.html
-Advertisement-
Play Games

在系統開發的時候一個老生常談的許可權管理問題,翻閱了很多的網路資料,但是總感覺離實際使用還有一段距離,其實許可權控制無非就幾個“請求、頁面按鈕、欄位顯示”, 對於前端許可權就需要配合JS了, 這裡主要展示我這些年來一直使用的一個關於許可權管理的案例,如果你只是寫一個簡單的CMS系統,那麼下麵代碼其實拷貝就可 ...


在系統開發的時候一個老生常談的許可權管理問題,翻閱了很多的網路資料,但是總感覺離實際使用還有一段距離,其實許可權控制無非就幾個“請求、頁面按鈕、欄位顯示”, 對於前端許可權就需要配合JS了, 這裡主要展示我這些年來一直使用的一個關於許可權管理的案例,如果你只是寫一個簡單的CMS系統,那麼下麵代碼其實拷貝就可以用了,如果是需要進行較大型系統開發就需要進一步封裝, 下麵是我的思路,寫的不好大家勿噴哈,有不同意見的留下一起探討。

1、在MVC開發時路由中包含了我們所有請求信息(Controller、Action、URL全文),而MVC又提供了過濾器的機制,因此我們將許可權管理封裝到過濾其中。首先把每個Controller的繼承封裝一個BaseController,這個Controller基類可以編寫一些直接調用的方法,例如Session等。

    /// <summary>
    /// 系統基礎控制器類
    /// </summary>
    public abstract class BaseUIController : Controller
    {
        /// <summary>
        /// 頁面初始化過程
        /// </summary>
        protected override void Initialize(System.Web.Routing.RequestContext requestContext)
        {
            requestContext.HttpContext.Response.Cache.SetCacheability(HttpCacheability.NoCache);
            base.Initialize(requestContext);
        }

        /// <summary>
        /// 內容返回
        /// </summary>
        /// <param name="content">特定內容供JS參考是否含流程定義等動作。</param>
        /// <param name="contentType"></param>
        /// <param name="contentEncoding"></param>
        /// <returns></returns>
        protected override ContentResult Content(string content, string contentType, Encoding contentEncoding)
        {
            return base.Content(content, contentType, contentEncoding);
        }

        /// <summary>
        /// 登陸用戶ID (Session存儲)
        /// </summary>
        public long LoginedUserID
        {
            get
            {
                object UID = Ant.UI.Controls.SessionConfig.GetSession("UID");
                if (Session == null || UID == null)
                    return -1;
                else
                {
                    return (long)UID;
                }
            }
            set
            {
                Ant.UI.Controls.SessionConfig.SetSession("UID", value);
            }
        }
        
        /// <summary>
        /// 獲取URL參數值,返回 NULL 或 值
        /// </summary>
        /// <param name="keyName">參數名</param>
        public string RequestUrlParams(string keyName)
        {
            if (Array.IndexOf(Request.QueryString.AllKeys, keyName) >= 0)
                return Request.QueryString[keyName].ToString();
            else
                return null;
        }
    }

2、這個時候我們編寫一個過濾器,在這個過濾器中可以在請求的Controller中獲取controller名稱和action名稱。 有了這兩個名稱就能定位到具體的頁面了,當然在得到頁面之前我們可以判斷用戶是否已經登錄。而頁面許可權頁已經用資料庫保存起來:A用戶或R角色對某一個Controller/Action的頁面(這個頁面可能是新增信息、修改、刪除信息的請求鏈接)是否有訪問許可權。

    /// <summary>
    /// 用戶登錄過濾器
    /// </summary>
    public class UserLoginedFilter : AuthorizeAttribute
    {
        public string name { get; set; }
        /// <summary>
        /// 頁面授權過程(驗證用戶登錄狀態)
        /// </summary>
        /// <param name="filterContext"></param>
        public override void OnAuthorization(AuthorizationContext filterContext)
        {
            UI.Controllers.BaseUIController _ControllerObj = (UI.Controllers.BaseUIController)filterContext.Controller;
            string controllerStr = _ControllerObj.ControllerContext.RouteData.Values["controller"].ToString();
            string actionStr = _ControllerObj.ControllerContext.RouteData.Values["action"].ToString();
            
            //用戶尚未登陸 或 登陸超時。
            if (_ControllerObj.LoginedUserID == -1)
            {
                UserError.LoginError();
            }
            // 許可權驗證 根據 Controller 與 Action(這裡可以將許可權緩存取出來進行用戶匹配,如果被拒絕可以拋出異常)
            else if ( controllerStr.ToLower() == "admin" && actionStr.ToLower() == "login")
            {
                throw new HttpException("許可權驗證。" + _ControllerObj.ControllerContext.Controller.ToString());
            }

            base.OnAuthorization(filterContext);
        }
    }

3、此時頁面調用時就不用在每個Action中編寫許可權驗證代碼啦(有些許可權驗證的源碼會在Action上增加過濾器,為了偷懶硬是想方設法把許可權封裝成直接在Controller類上添加一次過濾器就能完成的實現)

    [UserLoginedFilter]
    public class AdminController : UI.Controllers.BaseUIController
    {
        /// <summary>
        /// 登陸(登陸頁面不需要許可權驗證,增加AllowAnonymous特性)
        /// </summary>
        [AllowAnonymous]
        public ActionResult Login(FormCollection form)
        {
            return View();
        }

        /// <summary>
        /// 主界面
        /// </summary>
        public ActionResult Index()
        {
            return View();
        }
    }

上面只是我在系統開發過程中對於許可權管理的一個思路,也實實在在的應用到了一些系統中,當然上面這些代碼只能勉強判斷是否登陸,如果你需要對數據許可權、頁面許可權、欄位許可權進行管理那麼就需要進一步的擴展。如果需要在拋出異常後將異常信息友好的返回給客戶端,那你仍然需要在Global中捕獲返回狀態碼以及錯誤信息。因此我說這隻是我的一個思路,路過的朋友給我建議。多謝!

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 本案列僅針對剛剛入門vue學習的伙伴,博主也是剛剛在學基於vue cli搭建腳手架項目,對於前端大牛,可以移步。 快速搭建vue cli環境 如何搭建基於vue cli項目,這裡不再敘述,如果不會的伙伴可以自己百度。 項目搭建完成後,基本文件目錄如下: 打開我們剛剛創建的my project文 ...
  • 背景,做一個前面圖片寬度固定,後面寬度自適應,使用到了flex佈局,但是想讓後面div里文字不換行,超出以點點表示時,這時佈局就亂了,查了下,原來flex佈局與white-space:nowrap有影響 解決辦法,父div設置min-width:0即可 w3c上面是這樣說的 By default, ...
  • 三點註意事項 "JS作用域傳送門" JS沒有塊級作用域,只有全局作用域和局部作用域(函數作用域)。 JS中的作用域鏈,內部的作用域可以訪問到外部作用域中的變數和方法,而外部作用域不能訪問內部作用域的變數和方法。 當前作用域沒有此變數或方法,會向外部作用域尋找變數或方法。 閉包的兩種使用場景 函數作為 ...
  • clip-path介紹 clip-path 直譯過來就是裁剪路徑,使用SVG或形狀定義一個HTML元素的可見區域的方法。想象一下你在Photoshop中勾勒路徑的場景。MDN上是這樣介紹 clip-path的: clip-path屬性可以防止部分元素通過定義的剪切區域來顯示,僅通過顯示的特殊區域。剪 ...
  • 1 繪製文本 fillText(string,x,y,maxWidth) //填充試繪製文本 strokeText(string,x,y,maxWidth) 畫線試繪製文本 設置字體樣式: context.font 文字的字體樣式 可選屬性: font-style 字體樣式 font-variant ...
  • 一.概述 MVC的視圖與Razor頁面經常共用視覺和程式元素,通過使用佈局來完成,佈局還可減少重覆代碼。本章演示了以下內容的操作方法:(1)使用通用佈局,(2)自定義佈局,(3) 共用指令,(4)在呈現Razor頁面或MVC視圖之前運行通用代碼。 大多數 Web 應用都有一個通用佈局,可在頁面間切換 ...
  • 眾所周知C 提供Async和Await關鍵字來實現非同步編程。在本文中,我們將共同探討並介紹什麼是Async 和 Await,以及如何在C 中使用Async 和 Await。 同樣本文的內容也大多是翻譯的,只不過加上了自己的理解進行了相關知識點的補充,如果你認為自己的英文水平還不錯,大可直接跳轉到文章 ...
  • 前言: 發現這個問題,有不少人提起過,所以就簡單寫成文章吧。 接下來看如何在Aries 框架中使用存儲過程,整體步驟和綁定普通視圖差不多。 步驟一:新建一個空視圖。 可以在SqlCode管理中,創建一個空的視圖,主要用於顯示的表結構: 所以select 的空欄位名,和最終存儲過程出來的欄位名一致即可 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...