AspNetCore.Docs/aspnetcore/fundamentals/middleware.md

261 lines
15 KiB
Markdown
Raw Normal View History

---
2017-07-01 07:47:15 +08:00
title: ASP.NET Core Middleware
author: rick-anderson
description: Learn about ASP.NET Core middleware and the request pipeline.
keywords: ASP.NET Core,Middleware,pipeline,delegate
2016-10-29 01:35:15 +08:00
ms.author: riande
manager: wpickett
ms.date: 10/14/2017
2016-10-29 01:35:15 +08:00
ms.topic: article
ms.assetid: db9a86ab-46c2-40e0-baed-86e38c16af1f
2016-11-17 08:24:57 +08:00
ms.technology: aspnet
ms.prod: asp.net-core
uid: fundamentals/middleware
2016-10-29 01:35:15 +08:00
---
# ASP.NET Core Middleware Fundamentals
2016-10-29 01:35:15 +08:00
2017-10-14 04:50:30 +08:00
<a name="fundamentals-middleware"></a>
2016-10-29 01:35:15 +08:00
By [Rick Anderson](https://twitter.com/RickAndMSFT) and [Steve Smith](https://ardalis.com/)
2016-10-29 01:35:15 +08:00
[View or download sample code](https://github.com/aspnet/Docs/tree/master/aspnetcore/fundamentals/middleware/sample) ([how to download](xref:tutorials/index#how-to-download-a-sample))
2016-10-29 01:35:15 +08:00
## What is middleware
Middleware is software that is assembled into an application pipeline to handle requests and responses. Each component:
2016-10-29 01:35:15 +08:00
* Chooses whether to pass the request to the next component in the pipeline.
* Can perform work before and after the next component in the pipeline is invoked.
Request delegates are used to build the request pipeline. The request delegates handle each HTTP request.
Request delegates are configured using [Run](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.runextensions), [Map](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.mapextensions), and [Use](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.useextensions) extension methods. An individual request delegate can be specified in-line as an anonymous method (called in-line middleware), or it can be defined in a reusable class. These reusable classes and in-line anonymous methods are *middleware*, or *middleware components*. Each middleware component in the request pipeline is responsible for invoking the next component in the pipeline, or short-circuiting the chain if appropriate.
2016-10-29 01:35:15 +08:00
[Migrating HTTP Modules to Middleware](../migration/http-modules.md) explains the difference between request pipelines in ASP.NET Core and the previous versions and provides more middleware samples.
## Creating a middleware pipeline with IApplicationBuilder
2017-02-26 17:50:39 +08:00
The ASP.NET Core request pipeline consists of a sequence of request delegates, called one after the other, as this diagram shows (the thread of execution follows the black arrows):
2016-10-29 01:35:15 +08:00
![Request processing pattern showing a request arriving, processing through three middlewares, and the response leaving the application. Each middleware runs its logic and hands off the request to the next middleware at the next() statement. After the third middleware processes the request, it's handed back through the prior two middlewares for additional processing after the next() statements each in turn before leaving the application as a response to the client.](middleware/_static/request-delegate-pipeline.png)
2016-10-29 01:35:15 +08:00
2017-09-24 00:25:34 +08:00
Each delegate can perform operations before and after the next delegate. A delegate can also decide to not pass a request to the next delegate, which is called short-circuiting the request pipeline. Short-circuiting is often desirable because it avoids unnecessary work. For example, the static file middleware can return a request for a static file and short-circuit the rest of the pipeline. Exception-handling delegates need to be called early in the pipeline, so they can catch exceptions that occur in later stages of the pipeline.
2016-10-29 01:35:15 +08:00
2017-02-26 17:50:39 +08:00
The simplest possible ASP.NET Core app sets up a single request delegate that handles all requests. This case doesn't include an actual request pipeline. Instead, a single anonymous function is called in response to every HTTP request.
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Middleware/Startup.cs)]
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
The first [app.Run](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.runextensions) delegate terminates the pipeline.
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
You can chain multiple request delegates together with [app.Use](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.useextensions). The `next` parameter represents the next delegate in the pipeline. (Remember that you can short-circuit the pipeline by *not* calling the *next* parameter.) You can typically perform actions both before and after the next delegate, as this example demonstrates:
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Chain/Startup.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
>[!WARNING]
> Do not call `next.Invoke` after the response has been sent to the client. Changes to `HttpResponse` after the response has started will throw an exception. For example, changes such as setting headers, status code, etc, will throw an exception. Writing to the response body after calling `next`:
> - May cause a protocol violation. For example, writing more than the stated `content-length`.
> - May corrupt the body format. For example, writing an HTML footer to a CSS file.
>
> [HttpResponse.HasStarted](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.http.features.httpresponsefeature#Microsoft_AspNetCore_Http_Features_HttpResponseFeature_HasStarted) is a useful hint to indicate if headers have been sent and/or the body has been written to.
2016-10-29 01:35:15 +08:00
## Ordering
2016-10-29 01:35:15 +08:00
2017-02-26 17:50:39 +08:00
The order that middleware components are added in the `Configure` method defines the order in which they are invoked on requests, and the reverse order for the response. This ordering is critical for security, performance, and functionality.
The Configure method (shown below) adds the following middleware components:
1. Exception/error handling
2. Static file server
3. Authentication
4. MVC
# [ASP.NET Core 2.x](#tab/aspnetcore2x)
```csharp
public void Configure(IApplicationBuilder app)
{
app.UseExceptionHandler("/Home/Error"); // Call first to catch exceptions
// thrown in the following middleware.
app.UseStaticFiles(); // Return static files and end pipeline.
app.UseAuthentication(); // Authenticate before you access
// secure resources.
app.UseMvcWithDefaultRoute(); // Add MVC to the request pipeline.
}
```
# [ASP.NET Core 1.x](#tab/aspnetcore1x)
```csharp
public void Configure(IApplicationBuilder app)
{
app.UseExceptionHandler("/Home/Error"); // Call first to catch exceptions
// thrown in the following middleware.
app.UseStaticFiles(); // Return static files and end pipeline.
app.UseIdentity(); // Authenticate before you access
// secure resources.
app.UseMvcWithDefaultRoute(); // Add MVC to the request pipeline.
}
```
-----------
2017-02-26 17:50:39 +08:00
In the code above, `UseExceptionHandler` is the first middleware component added to the pipeline—therefore, it catches any exceptions that occur in later calls.
2017-02-26 17:50:39 +08:00
The static file middleware is called early in the pipeline so it can handle requests and short-circuit without going through the remaining components. The static file middleware provides **no** authorization checks. Any files served by it, including those under *wwwroot*, are publicly available. See [Working with static files](xref:fundamentals/static-files) for an approach to secure static files.
# [ASP.NET Core 2.x](#tab/aspnetcore2x)
If the request is not handled by the static file middleware, it's passed on to the Identity middleware (`app.UseAuthentication`), which performs authentication. Identity does not short-circuit unauthenticated requests. Although Identity authenticates requests, authorization (and rejection) occurs only after MVC selects a specific Razor Page or controller and action.
# [ASP.NET Core 1.x](#tab/aspnetcore1x)
2017-02-26 17:50:39 +08:00
If the request is not handled by the static file middleware, it's passed on to the Identity middleware (`app.UseIdentity`), which performs authentication. Identity does not short-circuit unauthenticated requests. Although Identity authenticates requests, authorization (and rejection) occurs only after MVC selects a specific controller and action.
-----------
2017-07-11 02:20:50 +08:00
The following example demonstrates a middleware ordering where requests for static files are handled by the static file middleware before the response compression middleware. Static files are not compressed with this ordering of the middleware. The MVC responses from [UseMvcWithDefaultRoute](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.mvcapplicationbuilderextensions#Microsoft_AspNetCore_Builder_MvcApplicationBuilderExtensions_UseMvcWithDefaultRoute_Microsoft_AspNetCore_Builder_IApplicationBuilder_) can be compressed.
```csharp
public void Configure(IApplicationBuilder app)
{
app.UseStaticFiles(); // Static files not compressed
// by middleware.
app.UseResponseCompression();
app.UseMvcWithDefaultRoute();
}
```
2017-10-14 04:50:30 +08:00
<a name="middleware-run-map-use"></a>
2016-10-29 01:35:15 +08:00
### Use, Run, and Map
2016-10-29 01:35:15 +08:00
You configure the HTTP pipeline using `Use`, `Run`, and `Map`. The `Use` method can short-circuit the pipeline (that is, if it does not call a `next` request delegate). `Run` is a convention, and some middleware components may expose `Run[Middleware]` methods that run at the end of the pipeline.
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
`Map*` extensions are used as a convention for branching the pipeline. [Map](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.mapextensions) branches the request pipeline based on matches of the given request path. If the request path starts with the given path, the branch is executed.
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Chain/StartupMap.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
2017-02-26 17:50:39 +08:00
The following table shows the requests and responses from `http://localhost:1234` using the previous code:
2016-10-29 01:35:15 +08:00
| Request | Response |
| --- | --- |
| localhost:1234 | Hello from non-Map delegate. |
| localhost:1234/map1 | Map Test 1 |
| localhost:1234/map2 | Map Test 2 |
| localhost:1234/map3 | Hello from non-Map delegate. |
2016-10-29 01:35:15 +08:00
When `Map` is used, the matched path segment(s) are removed from `HttpRequest.Path` and appended to `HttpRequest.PathBase` for each request.
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
[MapWhen](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.mapwhenextensions) branches the request pipeline based on the result of the given predicate. Any predicate of type `Func<HttpContext, bool>` can be used to map requests to a new branch of the pipeline. In the following example, a predicate is used to detect the presence of a query string variable `branch`:
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Chain/StartupMapWhen.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
2017-02-26 17:50:39 +08:00
The following table shows the requests and responses from `http://localhost:1234` using the previous code:
2016-10-29 01:35:15 +08:00
| Request | Response |
| --- | --- |
| localhost:1234 | Hello from non-Map delegate. |
| localhost:1234/?branch=master | Branch used = master|
2016-10-29 01:35:15 +08:00
`Map` supports nesting, for example:
```csharp
2016-10-29 01:35:15 +08:00
app.Map("/level1", level1App => {
level1App.Map("/level2a", level2AApp => {
// "/level1/level2a"
//...
});
level1App.Map("/level2b", level2BApp => {
// "/level1/level2b"
//...
});
});
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
`Map` can also match multiple segments at once, for example:
```csharp
app.Map("/level1/level2", HandleMultiSeg);
```
2016-10-29 01:35:15 +08:00
## Built-in middleware
ASP.NET Core ships with the following middleware components:
2016-10-29 01:35:15 +08:00
2016-12-14 14:17:38 +08:00
| Middleware | Description |
| ----- | ------- |
| [Authentication](xref:security/authentication/identity) | Provides authentication support. |
| [CORS](xref:security/cors) | Configures Cross-Origin Resource Sharing. |
| [Response Caching](xref:performance/caching/middleware) | Provides support for caching responses. |
| [Response Compression](xref:performance/response-compression) | Provides support for compressing responses. |
2016-12-14 14:17:38 +08:00
| [Routing](xref:fundamentals/routing) | Defines and constrains request routes. |
| [Session](xref:fundamentals/app-state) | Provides support for managing user sessions. |
| [Static Files](xref:fundamentals/static-files) | Provides support for serving static files and directory browsing. |
| [URL Rewriting Middleware](xref:fundamentals/url-rewriting) | Provides support for rewriting URLs and redirecting requests. |
2016-10-29 01:35:15 +08:00
2017-10-14 04:50:30 +08:00
<a name="middleware-writing-middleware"></a>
2016-10-29 01:35:15 +08:00
## Writing middleware
Middleware is generally encapsulated in a class and exposed with an extension method. Consider the following middleware, which sets the culture for the current request from the query string:
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Culture/StartupCulture.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
Note: The sample code above is used to demonstrate creating a middleware component. See [
Globalization and localization](xref:fundamentals/localization) for ASP.NET Core's built-in localization support.
2016-10-29 01:35:15 +08:00
You can test the middleware by passing in the culture, for example `http://localhost:7997/?culture=no`.
2016-10-29 01:35:15 +08:00
The following code moves the middleware delegate to a class:
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Culture/RequestCultureMiddleware.cs)]
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
The following extension method exposes the middleware through [IApplicationBuilder](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.iapplicationbuilder):
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Culture/RequestCultureMiddlewareExtensions.cs)]
2016-10-29 01:35:15 +08:00
The following code calls the middleware from `Configure`:
2016-10-29 01:35:15 +08:00
[!code-csharp[Main](middleware/sample/Culture/Startup.cs?name=snippet1&highlight=5)]
2016-10-29 01:35:15 +08:00
Middleware should follow the [Explicit Dependencies Principle](http://deviq.com/explicit-dependencies-principle/) by exposing its dependencies in its constructor. Middleware is constructed once per *application lifetime*. See *Per-request dependencies* below if you need to share services with middleware within a request.
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
Middleware components can resolve their dependencies from dependency injection through constructor parameters. [`UseMiddleware<T>`](https://docs.microsoft.com/aspnet/core/api/microsoft.aspnetcore.builder.usemiddlewareextensions#methods_summary) can also accept additional parameters directly.
2016-10-29 01:35:15 +08:00
### Per-request dependencies
2017-02-26 17:50:39 +08:00
Because middleware is constructed at app startup, not per-request, *scoped* lifetime services used by middleware constructors are not shared with other dependency-injected types during each request. If you must share a *scoped* service between your middleware and other types, add these services to the `Invoke` method's signature. The `Invoke` method can accept additional parameters that are populated by dependency injection. For example:
```c#
public class MyMiddleware
{
private readonly RequestDelegate _next;
public MyMiddleware(RequestDelegate next)
{
_next = next;
}
public async Task Invoke(HttpContext httpContext, IMyScopedService svc)
{
svc.MyProperty = 1000;
await _next(httpContext);
}
}
```
## Resources
2016-10-29 01:35:15 +08:00
* [Sample code used in this doc](https://github.com/aspnet/Docs/tree/master/aspnetcore/fundamentals/middleware/sample)
2016-10-29 01:35:15 +08:00
* [Migrating HTTP Modules to Middleware](../migration/http-modules.md)
* [Application Startup](startup.md)
* [Request Features](request-features.md)