1. 程式人生 > >asp.net core 系列 15 中間件

asp.net core 系列 15 中間件

set 負責 soft returns 生成 下一個 才會 n) private

原文:asp.net core 系列 15 中間件

一.概述

  中間件(也叫中間件組件)是一種裝配到應用管道以處理請求和響應的軟件。 每個組件:(1)選擇是否將請求傳遞到管道中的下一個組件;(2)可以在管道中的下一個組件之前和之後執行工作。

  請求委托用於生成請求管道。 請求委托會處理每個 HTTP 請求。使用以下方法配置請求委托:Run, Map, Use擴展方法。可以將單個請求委托作為匿名方法(稱為內聯中間件in-line middleware) 或者可以在可重用類中定義。這些可重用的類和內聯匿名方法是中間件,也稱為中間件組件。請求管道中的每個中間件組件負責調用管道中的下一個組件,或使管道短路。

  (1) Run

      //將終端中間件委托添加到應用程序的請求管道中。
      public static class RunExtensions
      {
          public static void Run(this IApplicationBuilder app, RequestDelegate handler);
      }

   (2) Map

      // 根據給定請求路徑的匹配對請求管道進行分支。
      public static class MapExtensions
      {
          
public static IApplicationBuilder Map(this IApplicationBuilder app, PathString pathMatch, Action<IApplicationBuilder> configuration);   }

  (3) Use

      // 提供配置應用程序請求的機制    
      public interface IApplicationBuilder
      {
          //....
          // 將中間件委托添加到應用程序的請求管道中。
          IApplicationBuilder Use(Func<RequestDelegate, RequestDelegate> middleware); 
      }

  1.1 使用 IApplicationBuilder 創建中間件管道

    在Startup. Configure方法中,使用IApplicationBuilder來創建中間件管理。每一個use開頭的擴展方法將一個中間件添加到IApplicationBuilder請求管道中。使用Use擴展方法來配置請求委托。每個use的中間件類似如下聲明:

    public static IApplicationBuilder Use[Middleware] (this IApplicationBuilder app )
      public static IApplicationBuilder Use[Middleware] (this IApplicationBuilder app , Action<T>)

    ASP.NET Core 請求管道包含一系列請求委托,依次調用。 下圖演示了這一概念。 沿黑色箭頭執行。

技術分享圖片

  在Startup. Configure代碼中,一系列use請求委托中間件如下所示:

           app.UseHttpsRedirection();
           app.UseStaticFiles();
           app.UseCookiePolicy();
           app.UseMvc();

    委托可以決定不將請求傳遞給下一個委托(中間件),這就是對請求管道進行短路。通常需要短路,因為這樣可以避免不必要的工作。

    下面示例 是一個最簡單的 ASP.NET core 應用程序,用run方法配置請求委托,設置單個委托處理處理所有請求。此案例不包括實際的請求管道。相反,調用單個匿名函數以響應每個 HTTP 請求。並用委托終止了管道。

    public class Startup
    {
        public void Configure(IApplicationBuilder app)
        {
            app.Run(async context =>
            {
                await context.Response.WriteAsync("Hello, World!");
            });
        }
    }

技術分享圖片

    下面示例用Use方法將多個請求委托鏈接在一起,next 參數表示管道中的下一個委托。 可通過不調用 next 參數使管道短路。

            app.Use(async (context, next) =>
            {
              //調用下一個委托(app.run)
                await next.Invoke();
            });

            app.Run(async context =>
            { 
                await context.Response.WriteAsync("Hello, World!");
            });

    

  1.2 中間件順序

    向 Startup.Configure 方法添加中間件組件的順序定義了針對請求調用這些組件的順序,以及響應的相反順序。 此排序對於安全性、性能和功能至關重要。以下 Startup.Configure 方法將為常見應用方案添加中間件組件:  

   (1) 異常/錯誤處理

   (2) HTTP 嚴格傳輸安全協議

  (3) HTTPS 重定向

  (4) 靜態文件服務器

   (5) Cookie 策略實施

  (6) 身份驗證

  (7) 會話

   (8) MVC

public void Configure(IApplicationBuilder app)
{
    if (env.IsDevelopment())
    {
        // When the app runs in the Development environment:
        //   Use the Developer Exception Page to report app runtime errors.
        //   Use the Database Error Page to report database runtime errors.
        app.UseDeveloperExceptionPage();
        app.UseDatabaseErrorPage();
    }
    else
    {
        // When the app doesn‘t run in the Development environment:
        //   Enable the Exception Handler Middleware to catch exceptions
        //     thrown in the following middlewares.
        //   Use the HTTP Strict Transport Security Protocol (HSTS)
        //     Middleware.
        app.UseExceptionHandler("/Error");
        app.UseHsts();
    }

    // Use HTTPS Redirection Middleware to redirect HTTP requests to HTTPS.
    app.UseHttpsRedirection();

    // Return static files and end the pipeline.
    app.UseStaticFiles();

    // Use Cookie Policy Middleware to conform to EU General Data 
    // Protection Regulation (GDPR) regulations.
    app.UseCookiePolicy();

    // Authenticate before the user accesses secure resources.
    app.UseAuthentication();

    // If the app uses session state, call Session Middleware after Cookie 
    // Policy Middleware and before MVC Middleware.
    app.UseSession();

    // Add MVC to the request pipeline.
    app.UseMvc();
}

    (1) UseExceptionHandler 是添加到管道的第一個中間件組件。 該異常處理程序中間件可捕獲稍後調用中發生的任何異常。

    (2) UseStaticFiles 靜態文件中間件,應該在管道的早期調用。這樣它就可以處理請求和短路,而不需要遍歷其余組件。靜態文件中間件不提供授權檢查。 它提供的任何文件,包括wwwroot下的文件,都是公開可訪問的。

    (3) UseAuthentication 身份驗證中間件。未經身份驗證的請求不會短路,但只有在特定的Razor頁面或MVC控制器操作之後,才會發生授權(和拒絕)。

  1.3 Use、Run 和 Map

    配置 HTTP 管道可以使用Use、Run 和 Map,但各方法針對構建的中間件作用不同:

      (1) Use[Middleware]中間件負責調用管道中的下一個中間件,也可使管道短路(即不調用 next 請求委托)。

      (2) Run[Middleware]是一種約定,一些中間件組件可能會公開在管道末端運行的Run[Middleware]方法。

      (3) Map擴展用作約定來創建管道分支, Map*創建請求管道分支是基於給定請求路徑的匹配項。

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
        {
            app.Map("/Map1",HandleMapTest1);
            app.Map("/Map2", HandleMapTest2);
            //其它請求地址
            app.Run(async context =>
            {
                await context.Response.WriteAsync("Hello from non-Map delegate. <p>");
            });
        }

        private static void HandleMapTest1(IApplicationBuilder app)
        {
             app.Run(async context =>
            {
                await context.Response.WriteAsync("Map Test 1");
            });
        }

        private static void HandleMapTest2(IApplicationBuilder app)
        {
            app.Run(async context =>
            {
                await context.Response.WriteAsync("Map Test 2");
            });
        }

技術分享圖片

    Map 還支持嵌套,下面的示例中,請求訪問/level1/level2a 和 /level1/level2b時進行不同邏輯處理:

app.Map("/level1", level1App => {
    level1App.Map("/level2a", level2AApp => {
        // "/level1/level2a" processing
    });
    level1App.Map("/level2b", level2BApp => {
        // "/level1/level2b" processing
    });
});

    

  1.4 MapWhen

    MapWhen 基於url給定謂詞的結果創建請求管道分支。 Func<HttpContext, bool> 類型的任何謂詞均可用於將請求映射到管道的新分支。 在以下示例中,謂詞用於檢測查詢字符串變量 branch 是否存在,如果存在使用新分支(HandleBranch)。

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
        {
            //Func<HttpContext, bool> predicate, Action<IApplicationBuilder> configuration
            app.MapWhen(context => context.Request.Query.ContainsKey("branch"), HandleBranch);
           
            //非匹配branch其它請求地址
            app.Run(async context =>
            {
                await context.Response.WriteAsync("Hello from non-Map delegate. <p>");
            });
        }

        private static void HandleBranch(IApplicationBuilder app)
        {
             app.Run(async context =>
            {
                var branchVer = context.Request.Query["branch"];
                await context.Response.WriteAsync("Map Test 1");
            });
        }

技術分享圖片

二. 編寫中間件    

  上面演示在請求管道中使用use,map,run方法,來委托處理每個 HTTP 請求就是中間件。通常中間件會封裝在類中,並且通過擴展方法公開。下面示例是如何編寫一個中間件組件。處理邏輯是該中間件通過查詢字符串設置當前請求的區域性。

    /// <summary>
    /// 自定義中間件實現類
    /// </summary>
    public class RequestCultureMiddleware
    {
        //using Microsoft.AspNetCore.Http
        private readonly RequestDelegate _next;

      
      /// <summary>
      /// 程序啟動時調用
      /// </summary>
      /// <param name="next"></param>
    public RequestCultureMiddleware(RequestDelegate next)
        {
            this._next = next;
        }

    
      /// <summary>
      ///每個頁面請求時自動調用,方法按約定命名,必需是Invoke或InvokeAsync
      /// </summary>
      /// <param name="context"></param>
      /// <returns></returns>
    public async Task InvokeAsync(HttpContext context)
        {
            var cultureQuery = context.Request.Query["culture"];
            if (!string.IsNullOrWhiteSpace(cultureQuery))
            {
                //using System.Globalization;
                var culture = new CultureInfo(cultureQuery);

                CultureInfo.CurrentCulture = culture;
                CultureInfo.CurrentUICulture = culture;

            }
            // Call the next delegate/middleware in the pipeline
            await _next(context);
        }
    }

    /// <summary>
    /// 通過擴展方法公開中間件 
    /// </summary>
    public static class RequestCultureMiddlewareExtensions
    {
        public static IApplicationBuilder UseRequestCulture(this IApplicationBuilder builder)
        {
            //在管道中添加一個use的中間件
            return builder.UseMiddleware<RequestCultureMiddleware>();
        }
    }
        public void Configure(IApplicationBuilder app)
        {
            //調用中間件
            app.UseRequestCulture();

            app.Run(async (context) =>
            {
                await ResponseAsync(context);
            });

        }

        private  async  Task ResponseAsync(HttpContext context)
        {
            context.Response.ContentType = "text/html; charset=utf-8";
            await context.Response.WriteAsync(
                    //打印當前顯示的語言
                    $"Hello { CultureInfo.CurrentCulture.DisplayName }"
                    );
        }

技術分享圖片

  2.1 請求依賴項

    由於中間件是在應用啟動時構造的(實例),而不是在每個請求時的,因此在每個請求過程中,中間件構造函數使用的作用域生命周期服務,不會在每個請求期間與其他依賴註入類型共享。如果必須在中間件和其他類型之間共享一個範圍服務,請將這些服務添加到 Invoke 方法的簽名。 Invoke 方法可接受由 DI 填充的參數

public class CustomMiddleware
{
    private readonly RequestDelegate _next;

    public CustomMiddleware(RequestDelegate next)
    {
        _next = next;
    }

    // IMyScopedService is injected into Invoke
    public async Task Invoke(HttpContext httpContext, IMyScopedService svc)
    {
        svc.MyProperty = 1000;
        await _next(httpContext);
    }
}

  參考文獻:

    官方文檔:ASP.NET Core 中間件

asp.net core 系列 15 中間件