18 KiB
title | author | description | monikerRange | ms.author | ms.custom | ms.date | uid |
---|---|---|---|---|---|---|---|
Policy-based authorization in ASP.NET Core | rick-anderson | Learn how to create and use authorization policy handlers for enforcing authorization requirements in an ASP.NET Core app. | >= aspnetcore-3.1 | riande | mvc | 1/5/2023 | security/authorization/policies |
Policy-based authorization in ASP.NET Core
:::moniker range=">= aspnetcore-6.0"
Underneath the covers, role-based authorization and claims-based authorization use a requirement, a requirement handler, and a preconfigured policy. These building blocks support the expression of authorization evaluations in code. The result is a richer, reusable, testable authorization structure.
An authorization policy consists of one or more requirements. Register it as part of the authorization service configuration, in the app's Program.cs
file:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Program.cs" range="20-23,29":::
In the preceding example, an "AtLeast21" policy is created. It has a single requirement—that of a minimum age, which is supplied as a parameter to the requirement.
IAuthorizationService
The primary service that determines if authorization is successful is xref:Microsoft.AspNetCore.Authorization.IAuthorizationService:
:::code language="csharp" source="~/security/authorization/policies/samples/stubs/copy_of_IAuthorizationService.cs" id="snippet" highlight="24-25,48-49":::
The preceding code highlights the two methods of the IAuthorizationService.
xref:Microsoft.AspNetCore.Authorization.IAuthorizationRequirement is a marker service with no methods, and the mechanism for tracking whether authorization is successful.
Each xref:Microsoft.AspNetCore.Authorization.IAuthorizationHandler is responsible for checking if requirements are met:
/// <summary>
/// Classes implementing this interface are able to make a decision if authorization
/// is allowed.
/// </summary>
public interface IAuthorizationHandler
{
/// <summary>
/// Makes a decision if authorization is allowed.
/// </summary>
/// <param name="context">The authorization information.</param>
Task HandleAsync(AuthorizationHandlerContext context);
}
The xref:Microsoft.AspNetCore.Authorization.AuthorizationHandlerContext class is what the handler uses to mark whether requirements have been met:
context.Succeed(requirement)
The following code shows the simplified (and annotated with comments) default implementation of the authorization service:
public async Task<AuthorizationResult> AuthorizeAsync(ClaimsPrincipal user,
object resource, IEnumerable<IAuthorizationRequirement> requirements)
{
// Create a tracking context from the authorization inputs.
var authContext = _contextFactory.CreateContext(requirements, user, resource);
// By default this returns an IEnumerable<IAuthorizationHandler> from DI.
var handlers = await _handlers.GetHandlersAsync(authContext);
// Invoke all handlers.
foreach (var handler in handlers)
{
await handler.HandleAsync(authContext);
}
// Check the context, by default success is when all requirements have been met.
return _evaluator.Evaluate(authContext);
}
The following code shows a typical authorization service configuration:
// Add all of your handlers to DI.
builder.Services.AddSingleton<IAuthorizationHandler, MyHandler1>();
// MyHandler2, ...
builder.Services.AddSingleton<IAuthorizationHandler, MyHandlerN>();
// Configure your policies
builder.Services.AddAuthorization(options =>
options.AddPolicy("Something",
policy => policy.RequireClaim("Permission", "CanViewPage", "CanViewAnything")));
Use xref:Microsoft.AspNetCore.Authorization.IAuthorizationService, [Authorize(Policy = "Something")]
, or RequireAuthorization("Something")
for authorization.
Apply policies to MVC controllers
For apps that use Razor Pages, see the Apply policies to Razor Pages section.
Apply policies to controllers by using the [Authorize]
attribute with the policy name:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Controllers/AtLeast21Controller.cs" id="snippet" highlight="1":::
If multiple policies are applied at the controller and action levels, all policies must pass before access is granted:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Controllers/AtLeast21Controller2.cs" id="snippet" highlight="1,4":::
Apply policies to Razor Pages
Apply policies to Razor Pages by using the [Authorize]
attribute with the policy name. For example:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Pages/AtLeast21.cshtml.cs" highlight="6":::
Policies can not be applied at the Razor Page handler level, they must be applied to the Page.
Policies can also be applied to Razor Pages by using an authorization convention.
Apply policies to endpoints
Apply policies to endpoints by using xref:Microsoft.AspNetCore.Builder.AuthorizationEndpointConventionBuilderExtensions.RequireAuthorization%2A with the policy name. For example:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Program.cs" id="snippet_requireAuthorization":::
Requirements
An authorization requirement is a collection of data parameters that a policy can use to evaluate the current user principal. In our "AtLeast21" policy, the requirement is a single parameter—the minimum age. A requirement implements xref:Microsoft.AspNetCore.Authorization.IAuthorizationRequirement, which is an empty marker interface. A parameterized minimum age requirement could be implemented as follows:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Requirements/MinimumAgeRequirement.cs":::
If an authorization policy contains multiple authorization requirements, all requirements must pass in order for the policy evaluation to succeed. In other words, multiple authorization requirements added to a single authorization policy are treated on an AND basis.
[!NOTE] A requirement doesn't need to have data or properties.
Authorization handlers
An authorization handler is responsible for the evaluation of a requirement's properties. The authorization handler evaluates the requirements against a provided xref:Microsoft.AspNetCore.Authorization.AuthorizationHandlerContext to determine if access is allowed.
A requirement can have multiple handlers. A handler may inherit xref:Microsoft.AspNetCore.Authorization.AuthorizationHandler%601, where TRequirement
is the requirement to be handled. Alternatively, a handler may implement xref:Microsoft.AspNetCore.Authorization.IAuthorizationHandler directly to handle more than one type of requirement.
Use a handler for one requirement
The following example shows a one-to-one relationship in which a minimum age handler handles a single requirement:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Handlers/MinimumAgeHandler.cs":::
The preceding code determines if the current user principal has a date of birth claim that has been issued by a known and trusted Issuer. Authorization can't occur when the claim is missing, in which case a completed task is returned. When a claim is present, the user's age is calculated. If the user meets the minimum age defined by the requirement, authorization is considered successful. When authorization is successful, context.Succeed
is invoked with the satisfied requirement as its sole parameter.
Use a handler for multiple requirements
The following example shows a one-to-many relationship in which a permission handler can handle three different types of requirements:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Handlers/PermissionHandler.cs":::
The preceding code traverses xref:Microsoft.AspNetCore.Authorization.AuthorizationHandlerContext.PendingRequirements%2A—a property containing requirements not marked as successful. For a ReadPermission
requirement, the user must be either an owner or a sponsor to access the requested resource. For an EditPermission
or DeletePermission
requirement, they must be an owner to access the requested resource.
Handler registration
Register handlers in the services collection during configuration. For example:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Program.cs" id="snippet_minimumAgeHandlerRegistration":::
The preceding code registers MinimumAgeHandler
as a singleton. Handlers can be registered using any of the built-in service lifetimes.
It's possible to bundle both a requirement and a handler into a single class implementing both xref:Microsoft.AspNetCore.Authorization.IAuthorizationRequirement and xref:Microsoft.AspNetCore.Authorization.IAuthorizationHandler. This bundling creates a tight coupling between the handler and requirement and is only recommended for simple requirements and handlers. Creating a class that implements both interfaces removes the need to register the handler in DI because of the built-in PassThroughAuthorizationHandler that allows requirements to handle themselves.
See the AssertionRequirement class for a good example where the AssertionRequirement
is both a requirement and the handler in a fully self-contained class.
What should a handler return?
Note that the Handle
method in the handler example returns no value. How is a status of either success or failure indicated?
-
A handler indicates success by calling
context.Succeed(IAuthorizationRequirement requirement)
, passing the requirement that has been successfully validated. -
A handler doesn't need to handle failures generally, as other handlers for the same requirement may succeed.
-
To guarantee failure, even if other requirement handlers succeed, call
context.Fail
.
If a handler calls context.Succeed
or context.Fail
, all other handlers are still called. This allows requirements to produce side effects, such as logging, which takes place even if another handler has successfully validated or failed a requirement. When set to false
, the xref:Microsoft.AspNetCore.Authorization.AuthorizationOptions.InvokeHandlersAfterFailure%2A property short-circuits the execution of handlers when context.Fail
is called. InvokeHandlersAfterFailure
defaults to true
, in which case all handlers are called.
[!NOTE] Authorization handlers are called even if authentication fails. Also handlers can execute in any order, so do not depend on them being called in any particular order.
Why would I want multiple handlers for a requirement?
In cases where you want evaluation to be on an OR basis, implement multiple handlers for a single requirement. For example, Microsoft has doors that only open with key cards. If you leave your key card at home, the receptionist prints a temporary sticker and opens the door for you. In this scenario, you'd have a single requirement, BuildingEntry, but multiple handlers, each one examining a single requirement.
BuildingEntryRequirement.cs
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Requirements/BuildingEntryRequirement.cs":::
BadgeEntryHandler.cs
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Handlers/BadgeEntryHandler.cs":::
TemporaryStickerHandler.cs
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Policies/Handlers/TemporaryStickerHandler.cs":::
Ensure that both handlers are registered. If either handler succeeds when a policy evaluates the BuildingEntryRequirement
, the policy evaluation succeeds.
Use a func to fulfill a policy
There may be situations in which fulfilling a policy is simple to express in code. It's possible to supply a Func<AuthorizationHandlerContext, bool>
when configuring a policy with the RequireAssertion
policy builder.
For example, the previous BadgeEntryHandler
could be rewritten as follows:
:::code language="csharp" source="~/security/authorization/policies/samples/6.0/AuthorizationPoliciesSample/Program.cs" range="20-21,25-29":::
Access MVC request context in handlers
The HandleRequirementAsync
method has two parameters: an AuthorizationHandlerContext
and the TRequirement
being handled. Frameworks such as MVC or SignalR are free to add any object to the Resource
property on the AuthorizationHandlerContext
to pass extra information.
When using endpoint routing, authorization is typically handled by the Authorization Middleware. In this case, the Resource
property is an instance of xref:Microsoft.AspNetCore.Http.HttpContext. The context can be used to access the current endpoint, which can be used to probe the underlying resource to which you're routing. For example:
if (context.Resource is HttpContext httpContext)
{
var endpoint = httpContext.GetEndpoint();
var actionDescriptor = endpoint.Metadata.GetMetadata<ControllerActionDescriptor>();
...
}
With traditional routing, or when authorization happens as part of MVC's authorization filter, the value of Resource
is an xref:Microsoft.AspNetCore.Mvc.Filters.AuthorizationFilterContext instance. This property provides access to HttpContext
, RouteData
, and everything else provided by MVC and Razor Pages.
The use of the Resource
property is framework-specific. Using information in the Resource
property limits your authorization policies to particular frameworks. Cast the Resource
property using the is
keyword, and then confirm the cast has succeeded to ensure your code doesn't crash with an InvalidCastException
when run on other frameworks:
// Requires the following import:
// using Microsoft.AspNetCore.Mvc.Filters;
if (context.Resource is AuthorizationFilterContext mvcContext)
{
// Examine MVC-specific things like routing data.
}
Globally require all users to be authenticated
Authorization with external service sample
The sample code on AspNetCore.Docs.Samples shows how to implement additional authorization requirements with an external authorization service. The sample Contoso.API
project is secured with Azure AD. An additional authorization check from the Contoso.Security.API
project returns a payload describing whether the Contoso.API
client app can invoke the GetWeather
API.
Configure the sample
-
Create an application registration in your Microsoft Entra ID tenant:
-
Assign it an AppRole.
-
Under API permissions, add the AppRole as a permission and grant Admin consent. Note that in this setup, this app registration represents both the API and the client invoking the API. If you like, you can create two app registrations. If you are using this setup, be sure to only perform the API permissions, add AppRole as a permission step for only the client. Only the client app registration requires a client secret to be generated.
-
Configure the
Contoso.API
project with the following settings:
:::code language="csharp" source="~/../AspNetCore.Docs.Samples/samples/aspnetcore-authz-with-ext-authz-service/Contoso.API/appsettings.json":::
- Configure
Contoso.Security.API
with the following settings:
:::code language="csharp" source="~/../AspNetCore.Docs.Samples/samples/aspnetcore-authz-with-ext-authz-service/Contoso.Security.API/appsettings.json":::
-
Open the ContosoAPI.collection.json file and configure an environment with the following:
ClientId
: Client Id from app registration representing the client calling the API.clientSecret
: Client Secret from app registration representing the client calling the API.TenantId
: Tenant Id from AAD properties
-
Extract the commands from the
ContosoAPI.collection.json
file and use them to construct cURL commands to test the app. -
Run the solution and use cURL to invoke the API. You can add breakpoints in the
Contoso.Security.API.SecurityPolicyController
and observe the client Id is being passed in that is used to assert whether it is allowed to Get Weather.
Additional resources
:::moniker-end