Authorization service
Describes the authorization service of the Optimizely Content Delivery API (version 2).
NoteThis section applies only if you are still using Content Delivery Api v2.x.x. For users using Content Delivery API v3.x.x, see API Authentication.
Optimizely Content Delivery API supports OAuth authentication (by installing the ContentDeliveryApi.OAuth package, see Install Content Delivery API and cookie-based authentication. However, ContentDeliveryApi exposes APIs that let you customize the authorization flow and use your preferred authorization mechanisms like AzureAD or GitHub.
NoteBy default, the
ContentDeliveryApi.OAuthpackage depends on ASP.NET Identity to work and it is not compatible with the old Membership provider.
The following diagram describes the Authorization flow of Content Delivery API:
-
When you request an endpoint (such as
api/episerver/v2.0/content/5), the request is handled byContentApiAuthorizationAttributebefore being passed to the corresponding controller. -
At
ContentApiAuthorizationAttribute, the flow is like the following:-
The user principal is initialized from
HttpActionContextusingISecurityPrincipal.InitializePrincipal(), and this principal is assigned to the current http context. -
The user principal is validated by
ContentApiAuthorizationService. There are two main steps atContentApiAuthorizationService:- The user principal is retrieved from
ISecurityPrincipal. - The principal is validated.
The final result should be a
Tuple<HttpStatusCode,string>that containsHttpStatusCodeand the error message if any errors occur. - The user principal is retrieved from
-
-
Based on the authorization result,
ContentApiAuthorizationAttributereturns an error or forwards the request to the controller endpoint.
ISecurityPrincipal and ContentApiAuthorizationService are important components for customizing the authorization flow:
ISecurityPrincipaldefines signatures for user principal manipulation. If you want to use another OAuth system, you should create your own implementation for this interface and define how to initialize and retrieve user principal:- Create a new class inheriting
ISecurityPrincipaland decorate the class with attributeServiceConfiguration(typeof(ISecurityPrincipal)). - Override the method
InitializePrincipalto initialize user claims. For example, a real-life scenario might be a customized OAuth system where the user’s access token is retrieved by calling one service, and another service is called for authentication and authorization. In this case, those operations should be put inInitializePrincipaland the final result should be that the principal for the current request is set (for example,HttpContext.Current.User = principalandThread.CurrentPrincipal = principal). - Implement
GetCurrentPrincipal()andGetAnonymousPrincipal()in line with your systems. - If your use case is complicated and this interface is not enough, you can customize the
Authorize()method ofContentApiAuthorizationService,
- Create a new class inheriting
This is the default implementation of ISecurityPrincipal:
using EPiServer.ServiceLocation;
using System.Security.Principal;
using System.Threading;
using System.Web;
using System.Web.Http.Controllers;
namespace EPiServer.ContentApi.Core.Security.Internal {
/// <summary>
/// Default implementation of <see cref="ISecurityPrincipal"/> for initialzing and accessing <see cref="IPrincipal"/> within ContentApi's scope.
/// </summary>
[ServiceConfiguration(typeof (ISecurityPrincipal))]
public class DefaultSecurityPrincipal: ISecurityPrincipal {
/// <summary>
/// Get <see cref="IPrincipal"/> from provided <see cref="HttpActionContext"/>. Set Principal on current thread and current <see cref="HttpContext"/>
/// </summary>
/// <param name="actionContext"></param>
public virtual void InitializePrincipal(HttpActionContext actionContext) {
// custom authentication logic, we must set the principal on two places: Current thread & current httpContext
// for reference, go here https://docs.microsoft.com/en-us/aspnet/web-api/overview/security/authentication-and-authorization-in-aspnet-web-api
Thread.CurrentPrincipal = actionContext.RequestContext.Principal;
// in case the application is not self-hosted
if (HttpContext.Current != null) {
HttpContext.Current.User = actionContext.RequestContext.Principal;
}
}
/// <inheritdoc />
public virtual IPrincipal GetCurrentPrincipal() {
return HttpContext.Current?.User ?? Thread.CurrentPrincipal;
}
/// <summary>
/// Retrieve an anonymous Principal with Anonymous and Everyone roles
/// </summary>
/// <returns></returns>
public virtual IPrincipal GetAnonymousPrincipal() {
return new GenericPrincipal(new GenericIdentity("Anonymous"), new [] {
"Everyone"
});
}
}
}ContentApiAuthorizationServiceis responsible for authorizing requests. It will check two main things: the user validity, and the user’s privileges to access the API. User principal is extracted byISecurityPrincipal.GetCurrentPrincipal(). If the process becomes complicated, this service should be customized. Otherwise,ISecurityPrincipalshould be enough.
See also: Content Delivery API and Custom Authorization blog post by Johan Kronberg
Updated 1 day ago