深入瞭解 Authorize 和 AllowAnonymous

来源:http://www.cnblogs.com/ben121011/archive/2016/08/29/5818923.html
-Advertisement-
Play Games

1.介紹 2.自定義 Filter V1.0 3.調試分析出現問題的原因 4.自定義 Filter V2.0 ...


深入瞭解 Authorize 和 AllowAnonymous

Chapter 0 - Intro

最近做的一個項目的時候,自定義授權 Attribute 來區分用戶許可權,我的項目不太大,許可權控制也不是很複雜,只涉及到匿名、普通用戶、超級管理員。 許可權驗證方式使用的是預設的 MemberShip 認證結合自己自定義的 許可權驗證 Filter。

Chapter 1 - 自定義 Filter V1.0

Filter代碼 V1.0

 1 /// <summary>
 2 /// 不需要登錄即可訪問
 3 /// </summary>
 4 public class NoPermissionRequiredAttribute : ActionFilterAttribute
 5 {
 6     public override void OnActionExecuting(ActionExecutingContext filterContext)
 7     {
 8         base.OnActionExecuting(filterContext);
 9     }
10 }
11 
12 /// <summary>
13 /// 需要登錄才能進行操作
14 /// </summary>
15 public class PermissionRequiredAttribute : ActionFilterAttribute
16 {
17     public override void OnActionExecuting(ActionExecutingContext filterContext)
18     {
19         if (filterContext.HttpContext.Session["User"]==null)
20         {
21             filterContext.Result = new RedirectResult("~/Admin/Account/Login");
22         }
23         base.OnActionExecuting(filterContext);
24     }
25 }
26 
27 /// <summary>
28 /// 需要有超級管理員許可權
29 /// </summary>
30 public class AdminPermissionRequiredAttribute : ActionFilterAttribute
31 {
32     public override void OnActionExecuting(ActionExecutingContext filterContext)
33     {
34         if ((filterContext.HttpContext.Session["User"] == null) || !((filterContext.HttpContext.Session["User"] as Models.User).IsSuper))
35         {
36             filterContext.Result = new RedirectResult("~/Admin/Account/Login");
37         }
38         base.OnActionExecuting(filterContext);
39     }
40 }

 

Chapter 2 - 控制器引用 Filter

控制器代碼中引用Filter 代碼:

 1 [Authorize]
 2 [Filters.PermissionRequired]
 3 public class AccountController : BaseAdminController
 4 {
 5     /// <summary>
 6     /// 登錄頁面
 7     /// </summary>
 8     /// <returns></returns>
 9     [AllowAnonymous]
10     [Filters.NoPermissionRequired]
11     [HttpGet]
12     public ActionResult Login(string ReturnUrl)
13     {
14         if (!Url.IsLocalUrl(ReturnUrl))
15         {
16             ReturnUrl = "/Admin/Home/Index";
17         }
18         if (Helpers.AuthFormService.TryAutoLogin())
19         {
20             return Redirect(ReturnUrl);
21         }
22         return View();
23     }
24 
25     /// <summary>
26     /// 賬戶首頁
27     /// </summary>
28     /// <returns></returns>
29     public ActionResult Index()
30     {
31         Models.User u = Session["User"] as Models.User;
32         return View(u);
33     }
34 }

 

Chapter 3 - 運行代碼

開始調試代碼,訪問這個 Login Action 時就直接崩了,一直在重定向到登錄頁面。 於是就想為什麼會出現這樣的情況呢,只使用 [Authorize][AllowAnonymous] 的時候是不會出現這種問題的, 但是為什麼自定義 Filter 的時候會出現這樣的問題呢,是哪裡出現的問題呢。
首先自帶的[Authorize][AllowAnonymous] 是基於 就近原則 的,離的越近的 Filter 的許可權越高會覆蓋掉父級定義的 Filter。 但是自定義的 Filter 卻並沒有依照 就近原則 這一原則來控制許可權,所以可能內部並不是靠判斷哪個 Filter 離得近就用哪個 Filter的,下一步反編譯 MVC 代碼,看 MVC 是怎麼樣處理 [Authorize][AllowAnonymous]

Chapter 4 - 反編譯分析出現問題的原因

利用 .Net 反編譯工具 Reflector 或 JustDecompile反編譯 System.Web.Mvc.dll ,在命名空間 System.Web.Mvc 下可以找到 AuthorizeAllowAnonymous的定義,如下圖所示:

system.web.mvc.png

AllowAnonymous定義:

AllowAnonymous.png

Authorize定義

Authorize.png

Authorize.OnAuthorization.png

通過上面的 Authorize.OnAuthorization 方法的定義基本可以知道問題出現在哪裡了,Authorize 在進行許可權驗證的時候會判斷當前請求的 Controller 和 Action 上是否有 AllowAnonymous 定義,如果有定義則不進行驗證,跳過驗證,只有在 當前請求的 Controller 和 Action 上都沒有 AllowAnonymous 時才會進行許可權驗證。

Chapter 5 - 自定義 Filter V2.0

知道問題出現在哪裡了,就開始修改自定義的 Filter 代碼吧,修改之後的代碼如下所示:

 1 /// <summary>
 2 /// 需要登錄才能進行操作
 3 /// </summary>
 4 public class PermissionRequiredAttribute : ActionFilterAttribute
 5 {
 6 
 7     public override void OnActionExecuting(ActionExecutingContext filterContext)
 8     {
 9         if (!filterContext.ActionDescriptor.IsDefined(typeof(NoPermissionRequiredAttribute),true))
10         {
11             if (filterContext.HttpContext.Session["User"] == null)
12             {
13                 filterContext.Result = new RedirectResult("~/Admin/Account/Login");
14             }
15         }            
16         base.OnActionExecuting(filterContext);
17     }
18 }

修改之後再次進行調試,導航到登錄頁面就不會再出現重定向的問題,就實現了按 就近原則 來決定 Filter 的優先順序的自定義的 Filter 了。


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

-Advertisement-
Play Games
更多相關文章
  • 1.MMU是Memory Management Unit的縮寫,中文名是記憶體管理單元,它是中央處理器(CPU)中用來管理虛擬存儲器、物理存儲器的控制線路,同時也負責虛擬地址映射為物理地址,以及提供硬體機制的記憶體訪問授權,多用戶多進程操作系統。 2.虛擬記憶體由來:許多年以前,當人們還在使用DOS或是更 ...
  • 文: 小波/QQ463431476 今年暑假參加了電子設計比賽,4天3夜(2016/7/25 7/29)選了個電子秤,所以想說說一點心得分享一下。秤可以測量,用了濾波和一個線性的方程。 硬體部分主要焊接4個電阻應變片搭建一個全橋電路,然後經過放大器,將電壓信號放大,HX711擁有一個24位 AD 來 ...
  • Linux下的文件許可權 在linux下每一個文件和目錄都有自己的訪問許可權,訪問許可權確定了用戶能否訪問文件或者目錄和怎樣進行訪問。最為我們熟知的一個文件或目錄可能擁有三種許可權,分別是讀、寫、和執行操作,在這裡不做詳細說明。我們創建一個文件後系統會預設地賦予所有者讀和寫許可權。當然我們也可以自己修改它,添 ...
  • 內容簡介:之前我們寫了一篇ESP8266AT指令的說明文檔,在客戶端連接上伺服器之後,接下來便是網路通信。為此我們又寫了這一片MQTT協議的說明文檔,更加深層次的講述了通信的原理。此文檔只是我們在調試過程中的一些小結和經驗,所以並不是規範的設計手冊。如果想要使用瞭解MQTT協議,一定是要仔細看到他的 ...
  • Microsoft Office Web Apps(以下簡稱OWA)是由微軟推出的基於Web端的線上辦公工具,它將Microsoft Office產品的體驗延伸到可支持的瀏覽器上。OWA讓你可以在任何地方共用自己的Office文檔。 ...
  • 1)讀取所有視圖: 2)讀取視圖的名稱: ...
  • hibernate 簡介:hibernate是一個開源框架,它是對象關聯關係映射的框架,它對JDBC做了輕量級的封裝,而我們java程式員可以使用面向對象的思想來操縱資料庫。hibernate核心介面session:負責被持久化對象CRUD操作sessionFactory:負責初始化hibernat ...
  • 【開源】.Net 免費開源,靜態Aop織入(直接修改IL中間語言)框架,類似PostSharp(收費);實現前後Aop切麵和INotifyPropertyChanged註入方式,性能接近原生編碼方式實現。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...