Jump to Section
This guide provides practical examples of Async Action Filters, including the recommended practices for injecting or dynamically resolving services.
What Are Async Action Filters?#
Async Action Filters are a type of filter in ASP.NET Core that let you inject logic around the execution of an action method:
- Before the Action Executes: Inspect or modify inputs, or prepare resources.
- After the Action Completes: Log results, modify responses, or clean up resources.
They are implemented using the OnActionExecutionAsync method, which wraps the execution of an action method.
The ASP.NET Core Request Pipeline
Filters operate within the MVC pipeline after middleware processes a request.
The diagram from Microsoft shows the execution order of filters.
Filters are executed in the following order:
- Authorization Filters: Validate access control.
- Resource Filters: Handle tasks like caching or resource initialization.
- Exception Filters: Handle unhandled exceptions.
- Action Filters: Execute logic before and after the action method.
- Result Filters: Modify the action result before sending it to the client.
Async Action Filters specifically wrap around the execution of action methods.
To understand how action filters work in ASP .NET Core - I encourage you to read Filters in ASP.NET Core on the MS website.
Async Action Filters Real-Life Examples#
1. Logging Requests and Responses
Log every incoming request and its corresponding response.
Register the Filter:
Apply the Filter:
2. Input Validation
Prevent invalid API requests from executing the action.
Register:
Apply:
3. Enriching Responses with Headers
Add a custom header to responses after the action executes.
Register:
Apply:
4. Dynamically Resolving Services
Dynamically retrieve services using RequestServices for conditionally required services.
Register:
Apply:
Async Action Filters Best Practices#
1. Call await next():
Ensure you call await next() to maintain the pipeline flow unless you intentionally short-circuited the request.
2. Keep Filters Lightweight:
Avoid heavy processing inside filters. Delegate complex logic to services to improve maintainability and performance.
3. Scope Filters Appropriately:
- Use global filters for shared concerns like logging.
- Use action-specific filters like input validation for localized logic.
4. Prefer Constructor Injection for Frequently Used Services:
Inject essential services directly into the filter’s constructor. This is the recommended approach for strong typing and easier testability.
5. Use RequestServices for Conditionally Required Services:
Dynamically resolve services using RequestServices only when they are optional or infrequently needed to avoid unnecessary dependencies.
Async Action Filters Common Pitfalls#
1. Overusing Filters for Global Concerns
Use middleware for application-wide tasks like authentication or error handling instead of filters.
2. Blocking Asynchronous Code
Avoid .Wait() or .Result in filters, as these can cause deadlocks. Always use await for asynchronous operations.
3. Forgetting to Register Filters
All filters used with [ServiceFilter] or [TypeFilter] must be registered in the DI container. Forgetting to register leads to runtime errors.
Conclusion#
Async Action Filters are a powerful way to add logic around controller actions, whether for logging, validation, or response customization.
Always register your filters in the DI container and ensure proper scoping for effective and maintainable implementations.