ASP.NET Web API自身對CORS的支持: EnableCorsAttribute特性背後的故事

来源:http://www.cnblogs.com/liuhui1/archive/2017/04/28/6782177.html
-Advertisement-
Play Games

從編程的角度來講,ASP.NET Web API針對CORS的實現僅僅涉及到HttpConfiguration的擴展方法EnableCors和EnableCorsAttribute特性。但是整個CORS體系不限於此,在它們背後隱藏著一系列的類型,我們將會利用本章餘下的內容對此作全面講述,今天我們就來 ...


從編程的角度來講,ASP.NET Web API針對CORS的實現僅僅涉及到HttpConfiguration的擴展方法EnableCors和EnableCorsAttribute特性。但是整個CORS體系不限於此,在它們背後隱藏著一系列的類型,我們將會利用本章餘下的內容對此作全面講述,今天我們就來討論一下用於定義CORS授權策略的EnableCorsAttribute特性背後的故事。

目錄
一、CorsPolicy
二、CorsPolicyProvider
三、CorsPolicyProviderFactory
四、CorsPolicyProviderFactory的註冊
五、總結

一、CorsPolicy

通過將EnableCorsAttribute特性應用到HttpController類型或者定義其中的某個Action方法上,我們可以為提供的資源定義相應的授權策略。ASP.NET Web API最終會利用這些策略對請求(包括預檢請求)進行解析並生成相應的CORS響應報頭。在ASP.NET Web API的應用編程介面中,CORS授權策略通過CorsPolicy類型表示。

通過《W3C的CORS規範》的介紹,我們知道針對跨域資源的授權策略不僅僅要求請求的源站點值得信任,還涉及到對請求採用的HTTP方法、攜帶的自定義報頭和用戶憑證的要求,以及針對自定義響應報頭的授權等。除此之外,為了避免頻繁瀏覽器頻繁地發送預檢請求,它可以將響應的結果進行緩存,而這又涉及到對緩存過期時間的控制。總得來說,這些授權策略體現在如下6個CORS響應報頭上。

  • Access-Control-Allow-Origin
  • Access-Control-Expose-Headers
  • Access-Control-Allow-Methods
  • Access-Control-Allow-Headers
  • Access-Control-Max-Age
  • Access-Control-Allow-Credentials

在ASP.NET Web API的應用編程介面中,圍繞著這6個CORS響應報頭的授權策略通過類型System.Web.Cors.CorsPolicy來表示。CorsPolicy具有如下6個屬性正好與上面這6個CORS響應報頭一一對應。

   1: public class CorsPolicy
   2: {
   3:     //其他成員
   4:     public IList<string>     Origins { get; }
   5:     public IList<string>     ExposedHeaders { get; }
   6:     public IList<string>     Headers { get; }
   7:     public IList<string>     Methods { get;; }
   8:     public long?             PreflightMaxAge { get; set; }
   9:     public bool              SupportsCredentials { get; set; }
  10: }

除了上述這6個屬性之外,CorsPolicy還具有如下3個布爾類型的屬性(AllowAnyOrigin、AllowAnyHeader和AllowAnyMethod),它們分別表示是否支持所有的源站點、自定義請求報頭和HTTP方法。

   1: public class CorsPolicy
   2: {
   3:     //其他成員
   4:     public bool AllowAnyOrigin { get; set; }
   5:     public bool AllowAnyHeader { get; set; }
   6:     public bool AllowAnyMethod { get; set; }
   7: }

 

二、CorsPolicyProvider

作為跨域資源請求進行授權檢查的依據,同時用於生成相應的CORS報頭的CorsPolicy對象通過另一個名為CorsPolicyProvider的對象來提供,所有的CorsPolicyProvider類型均實現了的介面System.Web.Http.Cors.ICorsPolicyProvider。如下麵的代碼片斷所示,該介面具有的唯一方法GetCorsPolicyAsync會根據代表但前請求的HttpRequestMessage對象得到表示CORS授權策略的CorsPolicy對象。

   1: public interface ICorsPolicyProvider
   2: {
   3:     Task<CorsPolicy> GetCorsPolicyAsync(HttpRequestMessage request, CancellationToken cancellationToken);
   4: }

實際上我們通過應用在目標HttpController類型或者定義其中的Action方法上用於定義CORS授權策略的System.Web.Http.Cors.EnableCorsAttribute就是ICorsPolicyProvider介面的實現者之一。如下麵的代碼片斷所示,EnableCorsAttribute同樣具有6個針對CORS響應報頭的屬性。在實現的GetCorsPolicyAsync方法中,它就是通過這6個屬性對返回的CorsPolicy對象進行初始化。

   1: [AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, AllowMultiple=false)]
   2: public sealed class EnableCorsAttribute : Attribute, ICorsPolicyProvider
   3: {    
   4:     public EnableCorsAttribute(string origins, string headers, string methods);
   5:     public EnableCorsAttribute(string origins, string headers, string methods,string exposedHeaders);
   6:  
   7:     public Task<CorsPolicy> GetCorsPolicyAsync(HttpRequestMessage request, CancellationToken cancellationToken);
   8:     
   9:     public IList<string> Origins { get; }
  10:     public IList<string> ExposedHeaders { get; }
  11:     public IList<string> Headers { get; }
  12:     public IList<string> Methods { get; }
  13:     public long          PreflightMaxAge { get; set; }
  14:     public bool          SupportsCredentials { get; set; }
  15: }

授權的源站點和允許的自定義請求報頭和HTTP方法,以及暴露給客戶端JavaScript程式的自定義響應報頭均可以直接通過構造函數參數來指定。對於這4個參數,我們可以指定某個單一的值(比如origin="http://www.artech.com"),也可以指定一個通過逗號分割的列表(比如origin="http://www.artech.com, http://www.jinnan.me")。除了exposedHeaders之外,我們還可以指定“*”作為其參數值,意味著不對此作任何限制,它們會控制生成CorsPolicy對象的3個對應布爾類型屬性值(AllowAnyOrigin、AllowAnyHeader和AllowAnyMethod)。

除了EnableCorsAttribute特性之外,在“System.Web.Http.Cors”命名空間下還定義著另一個與之相對的特性DisableCorsAttribute。顧名思義,如果DisableCorsAttribute特性被應用到某個HttpController類型或者定義其中的某個Action方法上,意味著目標HttpController或者Action不支持跨域資源共用。如下麵的代碼片斷所示,在實現的GetCorsPolicyAsync方法中,並沒有一個具體的CorsPolicy返回。

   1: [AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, AllowMultiple=false)]
   2: public sealed class DisableCorsAttribute : Attribute, ICorsPolicyProvider
   3: {
   4:     public Task<CorsPolicy> GetCorsPolicyAsync(HttpRequestMessage request, CancellationToken cancellationToken)
   5:     {
   6:         return Task.FromResult<CorsPolicy>(null);
   7:     }
   8: }

由於應用在Action方法上的CorsPolicyProvider特性比應用在HttpController類型上的特性具有更好的選擇優先順序,所以對於一個定義了眾多Action方法的HttpController類型來說,如果絕大部分Action方法均需要提供跨域資源共用的支持並具有相同的資源授權策略,可以直接在HttpController類型上應用EnableCorsAttribute特性並作相應的設置。對於不需要支持跨域資源共用的Action來說,直接在對應的方法上應用DisableCorsAttribute特性即可。如果某個Action具有特殊的授權需求,可以通過應用的EnableCorsAttribute特性作針對性設置。反之亦然。

三、CorsPolicyProviderFactory

CorsPolicyProvider用於提供用於描述CORS授權策略的CorsPolicy對象,其自身又通過對應的CorsPolicyProviderFactory來創建,所有的CorsPolicyProviderFactory類型均實現了介面System.Web.Http.Cors.ICorsPolicyProviderFactory。如下麵的代碼片斷所示,該介面具有的唯一方法GetCorsPolicyProvider會根據代表當前請求的HttpRequestMessage對象來提供對應的CorsPolicyProvider對象。

   1: public interface ICorsPolicyProviderFactory
   2: {
   3:     ICorsPolicyProvider GetCorsPolicyProvider(HttpRequestMessage request);
   4: }

由於提供的兩個具體CorsPolicyProvider類型(EnableCorsAttribute和DisableCorsAttribute)都是特性,所以ASP.NET Web API定義瞭如下一個AttributeBasedPolicyProviderFactory類型的CorsPolicyProviderFactory以解析特性的方式提供對應的CorsPolicyProvider。

   1: public class AttributeBasedPolicyProviderFactory : ICorsPolicyProviderFactory
   2: {    
   3:     public virtual ICorsPolicyProvider GetCorsPolicyProvider(HttpRequestMessage request);
   4:     public ICorsPolicyProvider DefaultPolicyProvider { get; set; }
   5: }

實現在GetCorsPolicyProvider方法中的CorsPolicyProvider提供機制很簡單:它直接利用註冊到當前ServicesContainer上的HttpActionSelector根據當前請求獲取用於描述目標Action的HttpActionDescriptor對象,然後調用其GetCustomAttributes<T>方法得到應用到對應Action方法上的第一個實現了ICorsPolicyProvider介面的特性。如果這樣的特性不存在,則獲取描述所在HttpController類型的HttpControllerDescritor對象,採用同樣的方式得到應用在目標HttpController類型上的第一個實現了ICorsPolicyProvider介面的特性。

關於針對目標Action的選擇問題,有一個核心的核心的細節值得關註:如果當前請求並非真正的跨域資源請求,而僅僅是一個採用“OPTIONS”作為HTTP方法的預檢請求(Preflight Request),利用註冊的HttpActionSelector根據當前請求是無法將目標Action選擇出來的,所以需要將請求的HTTP方法替換成真正跨域資源請求採用的HTTP方法。通過上面針對W3C的CORS規範的介紹我們知道,此HTTP方法可以通過預檢請求的“Access-Control-Request-Method”報頭獲得。實際上在上一個“通過自定義HttpMessageHandler實現CORS”的實例中,我們已經對此作個過演示了。

從上面給出的針對AttributeBasedPolicyProviderFactory的定義可以看出,除了實現的方法GetCorsPolicyProvider方法之外,它還具有一個DefaultPolicyProvider屬性。該屬性表示預設採用的CorsPolicyProvider,如果沒有任何實現ICorsPolicyProvider介面的特性被應用到目標Action方法和它所在的HttpController類型上,該屬性將會作為GetCorsPolicyProvider方法的返回值。

四、CorsPolicyProviderFactory的註冊

ASP.NET Web API預設使用的CorsPolicyProviderFactory需要註冊到當前的HttpConfiguration上。具體來說,所謂註冊CorsPolicyProviderFactory實際上就是將它保存到當前HttpConfiguration的Properties屬性表示的字典中。CorsPolicyProviderFactory的註冊可以通過HttpConfiguration如下所示的擴展方法SetCorsPolicyProviderFactory來完成。

另一擴展方法GetCorsPolicyProviderFactory 則用於獲取成功註冊的CorsPolicyProviderFactory。如果調用該方法CorsPolicyProviderFactory尚未被註冊,一個AttributeBasedPolicyProviderFactory對象會被創建出來並註冊到HttpConfiguration上。

   1: public static class CorsHttpConfigurationExtensions
   2: {
   3:     //其他成員
   4:     public static ICorsPolicyProviderFactory GetCorsPolicyProviderFactory(this HttpConfiguration httpConfiguration);
   5:     public static void SetCorsPolicyProviderFactory(this HttpConfiguration httpConfiguration, ICorsPolicyProviderFactory corsPolicyProviderFactory);
   6: }

五、總結

綜上所述,CorsPolicy用於描述具體的CORS資源授權策略,它由CorsPolicyProvider來提供,而後者又通過CorsPolicyProviderFactory來創建。如右圖所示的UML揭示了CorsPolicy、CorsPolicyProvider和CorsPolicyProviderFactory相關介面和類之間的關係。對於這些類型來說,除了CorsPolicy定義在程式集System.Web.Cors.dll,其餘的類型均定義在程式集System.Web.Http.Cors.dll中。

 

CORS系列文章
[1] 同源策略與JSONP
[2] 利用擴展讓ASP.NET Web API支持JSONP
[3] W3C的CORS規範
[4] 利用擴展讓ASP.NET Web API支持CORS
[5] ASP.NET Web API自身對CORS的支持: 從實例開始
[6] ASP.NET Web API自身對CORS的支持: CORS授權策略的定義和提供
[7] ASP.NET Web API自身對CORS的支持: CORS授權檢驗的實施
[8] ASP.NET Web API自身對CORS的支持: CorsMessageHandler

參考頁面:http://qingqingquege.cnblogs.com/p/5933752.html


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

-Advertisement-
Play Games
更多相關文章
  • 1 安裝tftp-server sudo apt-get install tftpd-hpa sudo apt-get install tftp-hpa(如果不需要客戶端可以不安裝) 2 配置tftp伺服器 第1步: 修改/etc/default/tftpd-hpa配置文件。 將/etc/defau ...
  • 最近有在學習會話共用的配置,其中一種呢是 nginx+redis+tomcat 的會話共用配置,在記錄此會話共用配置之前呢先記錄下redis等的安裝。這篇先簡單記錄下redis的安裝,是其中一種方式,其他的後續會進行補充或單獨記錄。 準備: 解壓tcl.tar.gz,並編譯安裝: 解壓redis.t ...
  • 以前簡單介紹過web api 的設計,但是還是有很多朋友問我,如何合理的設計和實現web api。比如,介面安全,異常處理,統一數據返回等問題。所以有必要系統的總結總結 web api 的設計和實現。由於前面已經介紹過web api 的參數和返回格式的設計,《Web API系列(一)設計經驗與總結》 ...
  • 一、前言 我們在優化Web服務的時候,對於靜態的資源文件,通常都是通過客戶端緩存、伺服器緩存、CDN緩存,這三種方式來緩解客戶端對於Web伺服器的連接請求壓力的。 本文指在這三個方面,在ASP.NET Core中靜態文件的實現過程和使用方法進行闡述。當然也可以考慮使用反向代理的方式(例如IIS或Ng ...
  • 這裡的“私闖sys.databases”是指Entity Framework預設發起的查詢:SELECT Count(*) FROM sys.databases WHERE [name]=N'資料庫名' 註:本文針對的是Entity Framework Code First場景,Entity Fra ...
  • 最近做一個指紋採集和比對的功能,因為公司整個項目是WEB類型的,所以指紋採集的模塊要嵌套在網頁中,那隻有用ActiveX了,以下是一些操作及效果,做個筆記! 新建用戶控制項,編寫CS代碼,如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 ...
  • Visual Studio 我們在Windows平臺上開發應用程式使用的工具主要是Visual Studio.這個集成開發環境已經演化了很多年,從一個簡單的C++編輯器和編譯器到一個高度集成、支持軟體開發整個生命周期的多語言環境。 Visual Studio以及它發佈的工具和服務提供了:設計、開發、 ...
  • 在開發Windows服務程式時,我們一般需要添加安裝程式,即:serviceInstaller,裡面有幾個關於名稱屬性,你都搞明白了嗎? 1.Description:表示服務說明(描述服務是乾什麼的); 2.DisplayName:表示友好名稱,可以理解為服務名的別名; 3.ServiceName: ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...