深入瞭解 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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...