探索ASP.NET Core中的IStartupFilter


原文:Exploring IStartupFilter in ASP.NET Core
作者:Andrew Lock
譯者:Lamond Lu

在本篇博客中,我將介紹一下IStartupFilter, 以及如何在ASP.NET Core中使用它。在下一篇博客中,我將介紹一下如何在外部中間件中使用IStartupFilter

IStartupFilter接口

IStartupFilter接口存在於Microsoft.AspNetCore.Hosting.Abstractions程序集中,它非常簡單,僅定義了一個接口方法。

namespace Microsoft.AspNetCore.Hosting
{
    public interface IStartupFilter
    {
        Action<IApplicationBuilder> Configure(Action<IApplicationBuilder> next);
    }
}

其中Configure方法返回了一個變量Action

當創建一個ASP.NET Core應用程序的時候,IApplicationBuilder負責配置ASP.NET Core的中間件管道。例如你可以在Startup.cs文件的Configure方法中,看到以下類似的代碼。

public void Configure(IApplicationBuilder app)
{
    app.UseStaticFiles();

    app.UseMvc(routes =>
    {
        routes.MapRoute(
            name: "default",
            template: "{controller=Home}/{action=Index}/{id?}");
    });
}

在這個方法中,你可以直接使用方法提供的IApplicationBuilder參數,並且可以向其中添加各種中間件。使用IStartupFilter, 你可以指定並返回一個Action 類型的泛型委托,這意味你除了可以使用方法提供的泛型委托配置IApplicationBuilder對象, 還需要返回一個泛型委托。

IStartupFilter方法可以接受一個配置IApplicationBuilder的方法,換而言之IStartupFilter.Configure方法可以使用Startup.Configure方法作為參數。

例:

Startup _startup = new Startup();
Action<IApplicationBuilder> startupConfigure = _startup.Configure;

//后續會補充StartupFilter1類的代碼
IStartupFilter filter1 = new StartupFilter1(); 

Action<IApplicationBuilder> filter1Configure = filter1.Configure(startupConfigure)

//后續會補充StartupFilter2類的代碼
IStartupFilter filter2 = new StartupFilter2(); 

Action<IApplicationBuilder> filter2Configure = filter2.Configure(filter1Configure)

如果之前你學習過ASP.NET Core的中間件管道,對於這個代碼,你可能會感覺很熟悉。這里我們正在建立另一條管道, 它是一個Configure方法的管道,而不是中間件管道。 這就是IStartupFilter的目的,允許在應用程序中創建Configure方法的管道。

實現IStartupFilter接口的對象何時會被調用?

現在我們對IStartupFilter的簽名有了更進一步的理解,接下來我們可以看看它在ASP.NET Core框架中的用法。

要查看IStartupFilter是如果被調用的,你可以在查看Microsoft.AspNetCore.Hosting程序集中的WebHost類。 當你在WebHostBuilder對象上調用Build方法時,實現IStartupFilter接口對象會被調用。 這個代碼通常出現在Program.cs文件中,例如:

public class Program
{
    public static void Main(string[] args)
    {
        var host = new WebHostBuilder()
            .UseKestrel()    
            .UseContentRoot(Directory.GetCurrentDirectory())
            .UseStartup<Startup>()
            .Build();  // 這個會調用BuildApplication方法

        host.Run(); 
    }
}

下面是BuildApplication方法的部分代碼,你可以看到這個方法負責初始化中間件管道。方法的返回值RequestDelegate表示了一個完整的管道,當請求到達的時候,Kestral服務器可以調用它。

private RequestDelegate BuildApplication()
{
    ..
    IApplicationBuilder builder = builderFactory.CreateBuilder(Server.Features);
    builder.ApplicationServices = _applicationServices;

    var startupFilters = _applicationServices.GetService<IEnumerable<IStartupFilter>>();
    Action<IApplicationBuilder> configure = _startup.Configure;
    foreach (var filter in startupFilters.Reverse())
    {
        configure = filter.Configure(configure);
    }

    configure(builder);

    return builder.Build();
}

首先,此方法創建IApplicationBuilder的實例,該實例將用於構建中間件管道,並將ApplicationServices設置為已配置的DI容器。

接下來的代碼塊很意思。首先,從DI容器中獲取了一個集合IEnumerable<IStartupFilter。正如我前面說的那樣,我們可以配置多個IStartupFilter來形成一個管道,所以這個方法只是從容器中取出它們。此外,Startup.Configure方法被保存到局部變量configure中, 這就是通常在Startup類中編寫的Configure方法,用於配置中間件管道。

現在我們通過循環遍歷每個IStartupFilter(以相反的順序),傳入Startup.Configure方法,然后更新局部變量configure來創建Configure方法的管道。這種方式實現了一種嵌套管道的效果。例如,如果我們有三個IStartupFilter實例,你最終會得到類似這樣的東西,其中內部Configure方法在參數中傳遞給外部方法:

局部變量configure的最終值會被IApplicationBuilder調用來執行實際的中間件管道配置。 調用builder.Build方法之后會生成處理HTTP請求所需的RequestDelegate

一個IStartupFilter的例子

前面我雖然描述了IStartupFilter的用途,但是可能查看一些現成的實現會更容易理解一些。 默認情況下,WebHostBuilder在初始化時會注冊一個IStartupFilter - AutoRequestServicesStartupFilter

public class AutoRequestServicesStartupFilter : IStartupFilter
{
    public Action<IApplicationBuilder> Configure(Action<IApplicationBuilder> next)
    {
        return builder =>
        {
            builder.UseMiddleware<RequestServicesContainerMiddleware>();
            next(builder);
        };
    }
}

本質上,它在中間件管道的開頭添加了一個額外的中間件,即RequestServicesContainerMiddleware

這是唯一一個默認注冊的IStartupFilter,因此在這種情況下,參數next將是Startup類的Configure方法。

這基本上就是IStartupFilter的全部內容 - 它是一種在配置的管道的開頭或結尾添加額外中間件(或其他配置)的方法。

如何注冊IStartupFilter

注冊IStartupFilter很簡單,只需像往常一樣在你的ConfigureServices方法中注冊它。 默認情況下,在WebHostBuilder中已經注冊了AutoRequestServicesStartupFilter

private IServiceCollection BuildHostingServices()
{
    ...
    services.AddTransient<IStartupFilter, AutoRequestServicesStartupFilter>();
    ...
}

RequestServicesContainerMiddleware中間件

以下是RequestServicesContainerMiddleware的部分代碼

public class RequestServicesContainerMiddleware
{
    private readonly RequestDelegate _next;
    private IServiceScopeFactory _scopeFactory;

    public RequestServicesContainerMiddleware(RequestDelegate next, IServiceScopeFactory scopeFactory)
    {
        _scopeFactory = scopeFactory;
        _next = next;
    }

    public async Task Invoke(HttpContext httpContext)
    {
        var existingFeature = httpContext.Features.Get<IServiceProvidersFeature>();

        if (existingFeature?.RequestServices != null)
        {
            await _next.Invoke(httpContext);
            return;
        }

        using (var feature = new RequestServicesFeature(_scopeFactory))
        {
            try
            {
                httpContext.Features.Set<IServiceProvidersFeature>(feature);
                await _next.Invoke(httpContext);
            }
            finally
            {
                httpContext.Features.Set(existingFeature);
            }
        }
    }
}

該中間件負責設置IServiceProvidersFeature。 創建時,RequestServicesFeature為請求創建新的IServiceScopeIServiceProvider。 它將負責使用Scoped生命周期添加到DI容器的依賴項的創建和處理。

IStartupFilter的使用場景

一般來說,我不認為在用戶的應用程序中需要使用IStartupFilter。 就其本質而言,用戶可以在Configure方法中定義中間件管道,因此IStartupFilter是不必要的。

我能想到以下幾種需要使用IStartupFilter的場景:

  • 你自己創建了一個庫,你需要確保你的中間件在中間件管道的開頭(或結尾)運行。
  • 你正在使用一個使用IStartupFilter的庫,您需要確保您的中間件在它之前運行。

總結

本篇博文中,我講解了IStartupFilter以及WebHost如何使用它在構建中間件管道。 在下一篇文章中,我將探討IStartupFilter的具體用法。

后記

本篇是作者早期的一篇博文,個人覺着對IStartupFilter講解的比較清楚,就翻譯了一下。在作者的后期博文中,作者提供了許多IStartupFilter的使用場景,例如

有興趣的同學可以自己閱讀一下,后續我會選擇一些有意思的文章翻譯一下。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM