# [Filters](https://docs.microsoft.com/en-us/aspnet/core/mvc/controllers/filters) **Filters** in ASP.NET Core allow code to be run *before* or *after* specific stages in the request processing pipeline. Built-in filters handle tasks such as: - Authorization (preventing access to resources a user isn't authorized for). - Response caching (short-circuiting the request pipeline to return a cached response). Custom filters can be created to handle cross-cutting concerns. Examples of cross-cutting concerns include error handling, caching, configuration, authorization, and logging. Filters avoid duplicating code. ## **How filters work** Filters run within the *ASP.NET Core action invocation pipeline*, sometimes referred to as the *filter pipeline*. The filter pipeline runs after ASP.NET Core selects the action to execute. ![1][filter-pipeline-1] ![2][filter-pipeline-2] [filter-pipeline-1]: https://docs.microsoft.com/it-it/aspnet/core/mvc/controllers/filters/_static/filter-pipeline-1.png [filter-pipeline-2]: https://docs.microsoft.com/en-gb/aspnet/core/mvc/controllers/filters/_static/filter-pipeline-2.png ## **Filter types** Each filter type is executed at a different stage in the filter pipeline: - **Authorization filters** run first and are used to determine whether the user is authorized for the request. Authorization filters short-circuit the pipeline if the request is not authorized. - **Resource filters**: - Run after authorization. - `OnResourceExecuting` runs code before the rest of the filter pipeline. For example, `OnResourceExecuting` runs code before model binding. - `OnResourceExecuted` runs code after the rest of the pipeline has completed. - **Action filters**: - Run code immediately before and after an action method is called. - Can change the arguments passed into an action. - Can change the result returned from the action. - Are **not** supported in Razor Pages. - **Exception filters** apply global policies to unhandled exceptions that occur before the response body has been written to. - **Result filters** run code immediately before and after the execution of action results. They run only when the action method has executed successfully. They are useful for logic that must surround view or formatter execution. ## **Implementation** Filters support both synchronous and asynchronous implementations through different interface definitions. For example, `OnActionExecuting` is called before the action method is called. `OnActionExecuted` is called after the action method returns. Asynchronous filters define an `On-Stage-ExecutionAsync` method, for example `OnActionExecutionAsync`. Interfaces for multiple filter stages can be implemented in a single class. ## **Built-in filter attributes** ASP.NET Core includes built-in *attribute-based* filters that can be subclassed and customized. Several of the filter interfaces have corresponding attributes that can be used as base classes for custom implementations. Filter attributes: - [ActionFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.filters.actionfilterattribute) - [ExceptionFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.filters.exceptionfilterattribute) - [ResultFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.filters.resultfilterattribute) - [FormatFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.formatfilterattribute) - [ServiceFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.servicefilterattribute) - [TypeFilterAttribute](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.typefilterattribute) ## **Filter scopes** A filter can be added to the pipeline at one of three *scopes*: - Using an attribute on a controller action. Filter attributes cannot be applied to Razor Pages handler methods. ```cs // services.AddScoped(); [ServiceFilter(typeof(CustomActionFilterAttribute))] public IActionResult Index() { return Content("Header values by configuration."); } ``` - Using an attribute on a controller or Razor Page. ```cs // services.AddControllersWithViews(options => { options.Filters.Add(new CustomResponseFilterAttribute(args)); }); [CustomResponseFilterAttribute(args)] public class SampleController : Controller // or [CustomResponseFilterAttribute(args)] [ServiceFilter(typeof(CustomActionFilterAttribute))] public class IndexModel : PageModel ``` - Globally for all controllers, actions, and Razor Pages. ```cs public void ConfigureServices(IServiceCollection services) { services.AddControllersWithViews(options => { options.Filters.Add(typeof(CustomActionFilter)); }); } ``` ## Filter Order of Execution When there are multiple filters for a particular stage of the pipeline, scope determines the default order of filter execution. Global filters surround class filters, which in turn surround method filters. As a result of filter nesting, the *after* code of filters runs in the reverse order of the *before* code. The filter sequence: - The *before* code of global filters. - The *before* code of controller and Razor Page filters. - The *before* code of action method filters. - The *after* code of action method filters. - The *after* code of controller and Razor Page filters. - The *after* code of global filters. ### Cancellation and Short-Circuiting The filter pipeline can be short-circuited by setting the `Result` property on the `ResourceExecutingContext` parameter provided to the filter method. ```cs public class ShortCircuitingResourceFilterAttribute : Attribute, IResourceFilter { public void OnResourceExecuting(ResourceExecutingContext context) { context.Result = new ContentResult() { Content = "Resource unavailable - header not set." }; } public void OnResourceExecuted(ResourceExecutedContext context) { } } ```