Two-factor authentication (2FA) for Umbraco members is activated by implementing an ITwoFactorProvider interface and registering the implementation. The implementation can use third-party packages to support authentication apps like the Microsoft- or Google Authentication Apps.
The following guide will take you through implementing an option for your website members to enable two-factor authentication.
A setup for members needs to be implemented on your website in order for you to follow this guide. This setup should include:
Login and logout options.
Public access restriction configured on at least 1 content item.
As an example, the guide will use the GoogleAuthenticator NuGet Package. This package works for both Google and Microsoft authenticator apps. It can be used to generate the QR code needed to activate the app for the website.
Install the GoogleAuthenticator Nuget Package on your project.
Create a new file in your project: QrCodeSetupData.cs.
Update the file with the following code snippet.
QrCodeSetupData.cs
usingSystem;usingSystem.Threading.Tasks;usingGoogle.Authenticator;usingUmbraco.Cms.Core.Security;usingUmbraco.Cms.Core.Services;namespaceMy.Website;/// <summary>/// Model with the required data to setup the authentication app./// </summary>publicclassQrCodeSetupData{ /// <summary> /// The secret unique code for the user and this ITwoFactorProvider. /// </summary>publicstring? Secret { get; init; } /// <summary> /// The SetupCode from the GoogleAuthenticator code. /// </summary>publicSetupCode? SetupCode { get; init; }}/// <summary>/// App Authenticator implementation of the ITwoFactorProvider/// </summary>publicclassUmbracoAppAuthenticator:ITwoFactorProvider{ /// <summary> /// The unique name of the ITwoFactorProvider. This is saved in a constant for reusability. /// </summary>publicconststring Name ="UmbracoAppAuthenticator";privatereadonlyIMemberService _memberService; /// <summary> /// Initializes a new instance of the <seecref="UmbracoAppAuthenticator"/> class. /// </summary>publicUmbracoAppAuthenticator(IMemberService memberService) { _memberService = memberService; } /// <summary> /// The unique provider name of ITwoFactorProvider implementation. /// </summary> /// <remarks> /// This value will be saved in the database to connect the member with this ITwoFactorProvider. /// </remarks>publicstring ProviderName => Name; /// <summary> /// Returns the required data to setup this specific ITwoFactorProvider implementation. In this case it will contain the url to the QR-Code and the secret. /// </summary> /// <paramname="userOrMemberKey">The key of the user or member</param> /// <paramname="secret">The secret that ensures only this user can connect to the authenticator app</param> /// <returns>The required data to setup the authenticator app</returns>publicTask<object> GetSetupDataAsync(Guid userOrMemberKey,string secret) {var member =_memberService.GetByKey(userOrMemberKey);var applicationName ="My Application Name";var twoFactorAuthenticator =newTwoFactorAuthenticator();SetupCode setupInfo =twoFactorAuthenticator.GenerateSetupCode(applicationName,member.Username, secret,false);returnTask.FromResult<object>(newQrCodeSetupData() { SetupCode = setupInfo, Secret = secret }); } /// <summary> /// Validated the code and the secret of the user. /// </summary>publicboolValidateTwoFactorPIN(string secret,string code) {var twoFactorAuthenticator =newTwoFactorAuthenticator();returntwoFactorAuthenticator.ValidateTwoFactorPIN(secret, code); } /// <summary> /// Validated the two factor setup /// </summary> /// <remarks>Called to confirm the setup of two factor on the user. In this case we confirm in the same way as we login by validating the PIN.</remarks>publicboolValidateTwoFactorSetup(string secret,string token) => ValidateTwoFactorPIN(secret, token);}
Update namespace on line 7 to match your project.
Customize the applicationName variable on line 63.
Create a Composer and register the UmbracoAppAuthenticator implementation as shown below.
At this point, the 2FA is active, but no members have set up 2FA yet. The setup of 2FA depends on the type. In the case of App Authenticator, we will add the following to our view showing the edit profile of the member.
Add or choose a members-only page that should have the two-factor authentication setup.
The page needs to be behind the public access.
The page should not be using strongly types models.
Open the view file for the selected page.
Add the following code:
ExampleMembersPage.cshtml
@using Umbraco.Cms.Core.Services@using Umbraco.Cms.Web.Website.Controllers@using Umbraco.Cms.Web.Website.Models@using My.Website @* Or whatever your namespacewiththeQrCodeSetupDatamodelis *@@injectMemberModelBuilderFactorymemberModelBuilderFactory@injectITwoFactorLoginServicetwoFactorLoginService@{ // Build a profile model to edit, by fetching the member's unique key. var profileModel = await memberModelBuilderFactory .CreateProfileModel() .BuildForCurrentMemberAsync(); // Show all two factor providers var providerNames = twoFactorLoginService.GetAllProviderNames(); if (providerNames.Any()) { <div asp-validation-summary="All" class="text-danger"></div> foreach (var providerName in providerNames) {var setupData =awaittwoFactorLoginService.GetSetupInfoAsync(profileModel.Key, providerName); // If the `setupData` is `null` for the specified `providerName` it means the provider is already set up. // In this case, a button to disable the authentication is shown.if (setupData is null) { @using (Html.BeginUmbracoForm<UmbTwoFactorLoginController>(nameof(UmbTwoFactorLoginController.Disable))) {<input type="hidden" name="providerName" value="@providerName"/><button type="submit">Disable @providerName</button> } } // If `setupData` is not `null` the type is checked and the UI for how to set up the App Authenticator is shown.elseif(setupData is QrCodeSetupData qrCodeSetupData) { @using (Html.BeginUmbracoForm<UmbTwoFactorLoginController>(nameof(UmbTwoFactorLoginController.ValidateAndSaveSetup))) {<h3>Setup @providerName</h3><img src="@qrCodeSetupData.SetupCode.QrCodeSetupImageUrl"/><p>Scan the code above with your authenticator app <br /> and enter the resulting code here to validate:</p><input type="hidden" name="providerName" value="@providerName"/><input type="hidden" name="secret" value="@qrCodeSetupData.Secret"/><input type="text" name="code"/><button type="submit">Validate & save</button> } } } }}
Test the set up
Login to the website using a test member.
Navigate to the page where the QR code was added.
Scan the QR code and add the verification code.
Logout of the website.
Login and verify that it asks for the two factor authentication.
You can also check that the Two-factor Authentication option is checked on the member in the Umbraco backoffice.
Notification when 2FA is requested for a member
When a 2FA login is requested for a member, the MemberTwoFactorRequestedNotification is published. This notification can also be used to send the member a one-time password via e-mail or phone. Even though these 2FA types are not considered secure as App Authentication, it is still a massive improvement compared to no 2FA.
Two-factor authentication for Users
Umbraco controls how the UI is for user login and user edits, but will still need a view for configuring each 2FA provider.
Example implementation for Authenticator Apps for Users
In the following example, we will use the GoogleAuthenticator NuGet Package. Despite the name, this package works for both Google and Microsoft authenticator apps. It can be used to generate the QR code needed to activate the app for the website.
usingSystem.Runtime.Serialization;usingGoogle.Authenticator;usingUmbraco.Cms.Core.Models.Membership;usingUmbraco.Cms.Core.Security;usingUmbraco.Cms.Core.Services;namespaceMy.Website;[DataContract]publicclassTwoFactorAuthInfo{ [DataMember(Name ="qrCodeSetupImageUrl")]publicstring? QrCodeSetupImageUrl { get; set; } [DataMember(Name ="secret")]publicstring? Secret { get; set; }}/// <summary>/// App Authenticator implementation of the ITwoFactorProvider/// </summary>publicclassUmbracoUserAppAuthenticator:ITwoFactorProvider{privatereadonlyIUserService _userService; /// <summary> /// The unique name of the ITwoFactorProvider. This is saved in a constant for reusability. /// </summary>publicconststring Name ="UmbracoUserAppAuthenticator"; /// <summary> /// Initializes a new instance of the <seecref="UmbracoUserAppAuthenticator"/> class. /// </summary>publicUmbracoUserAppAuthenticator(IUserService userService) { _userService = userService; } /// <summary> /// The unique provider name of ITwoFactorProvider implementation. /// </summary> /// <remarks> /// This value will be saved in the database to connect the member with this ITwoFactorProvider. /// </remarks>publicstring ProviderName => Name; /// <summary> /// Returns the required data to setup this specific ITwoFactorProvider implementation. In this case it will contain the url to the QR-Code and the secret. /// </summary> /// <paramname="userOrMemberKey">The key of the user or member</param> /// <paramname="secret">The secret that ensures only this user can connect to the authenticator app</param> /// <returns>The required data to setup the authenticator app</returns>publicTask<object> GetSetupDataAsync(Guid userOrMemberKey,string secret) {IUser? user =_userService.GetByKey(userOrMemberKey);ArgumentNullException.ThrowIfNull(user);var twoFactorAuthenticator =newTwoFactorAuthenticator();SetupCode setupInfo =twoFactorAuthenticator.GenerateSetupCode("My application name",user.Username, secret,false);returnTask.FromResult<object>(newTwoFactorAuthInfo() { QrCodeSetupImageUrl =setupInfo.QrCodeSetupImageUrl, Secret = secret }); } /// <summary> /// Validated the code and the secret of the user. /// </summary>publicboolValidateTwoFactorPIN(string secret,string code) {var twoFactorAuthenticator =newTwoFactorAuthenticator();returntwoFactorAuthenticator.ValidateTwoFactorPIN(secret, code); } /// <summary> /// Validated the two factor setup /// </summary> /// <remarks>Called to confirm the setup of two factor on the user. In this case we confirm in the same way as we login by validating the PIN.</remarks>publicboolValidateTwoFactorSetup(string secret,string token) => ValidateTwoFactorPIN(secret, token);}
First, we create a model with the information required to set up the 2FA provider. Then we implement the ITwoFactorProvider with the use of the TwoFactorAuthenticator from the GoogleAuthenticator NuGet package.
Now we need to register the UmbracoUserAppAuthenticator implementation and the view to show to set up this provider. This can be done on the IUmbracoBuilder in your startup or a composer.
!(function () {"use strict";constgoogleTwoFactorProviderCtrl= ['$scope','twoFactorLoginResource','notificationsService',function ($scope, twoFactorLoginResource, notificationsService) {constvm=this;vm.title ="Setup Google Authenticator on "+$scope.model?.user?.name;vm.providerName =$scope.model?.providerName;vm.qrCodeImageUrl ="";vm.secret ="";vm.code ="";vm.authForm = {};vm.buttonState ="init";vm.close = close;vm.validateAndSave = validateAndSave;functioninit() {vm.buttonState ="init";twoFactorLoginResource.setupInfo(vm.providerName).then(function (response) {// This response is the model I defined to be returned from ITwoFactorProvider.GetSetupDataAsyncvm.qrCodeImageUrl =response.qrCodeSetupImageUrl;vm.secret =response.secret; }).catch(function () {notificationsService.error("Could not fetch login info"); }); }functionvalidateAndSave() {vm.authForm.token.$setValidity("token",true);vm.buttonState ="busy";twoFactorLoginResource.validateAndSave(vm.providerName,vm.secret,vm.code).then(function (successful) {if (successful) {notificationsService.success("Two-factor authentication has successfully been enabled");vm.buttonState ="success";close(); } else {vm.authForm.token.$setValidity("token",false);vm.buttonState ="error"; } }).catch(function (error) {notificationsService.error(error);vm.buttonState ="error"; }); }functionclose() {if ($scope.model.close) {$scope.model.close(); } }init(); } ];angular.module("umbraco").controller("CustomCode.TwoFactorProviderGoogleAuthenticator", googleTwoFactorProviderCtrl);})();
At this point, the 2FA is active, but no users have set up 2FA yet.
Each user can now enable the configured 2fa providers on their user. This can be done from the user panel by clicking the user avatar.
When clicking the Configure Two-Factor button, a new panel is shown, listing all enabled two-factor providers.
When clicking Enable on one of these, the configured view for the specific provider will be shown
When the authenticator is enabled correctly, a disable button is shown instead.
To disable the two-factor authentication on your user, it is required to enter the verification code. Otherwise, admins are allowed to disable providers on other users.
If the code is correct, the provider is disabled.
Notification when 2FA is requested for a user
When a 2FA login is requested for a user, the UserTwoFactorRequestedNotification is published. This notification can also be used to send the user a one-time password via e-mail or phone. Even though these 2FA types are not considered secure as App Authentication, it is still a massive improvement compared to no 2FA.
Login with 2FA enabled
When a user with 2FA enabled logs in, they will be presented with a screen to enter the verification code:
While the 2FA is enabled, the user will be presented with this screen after entering the username and password.
If the code is correct, the user will be logged in. If the code is incorrect, the user will be presented with an error message.
This screen is set up to work well with 2FA providers that require a one-time code to be entered. The code field follows best practices for accessibility in terms of labeling and autocompletion.
A user can have more than one 2FA provider activated simultaneously. In this case, the user will be presented with a dropdown to choose which provider to use before entering a code.
Customizing the login screen
The 2FA login screen can be customized. This should be done if you have a 2FA provider that does not require a one-time code to be entered.
You should only customize the 2FA login screen in certain cases, for example:
If you have a provider that requires a non-numeric field or additional info.
If you have a provider that requires the user to scan a QR code, you should additionally show the QR code.
If you need to authenticate the user in a different way than the default AuthenticationController in Umbraco.
You need to create a JavaScript module that exports a default custom element to be used in the login screen. This module should be placed in the App_Plugins folder. The module should be registered using a composer.
In earlier versions of Umbraco up to version 12, you had to define an AngularJS HTML view. This is no longer the case. You can now define a JavaScript module to render a Custom Element instead of the default two-factor login screen.
It is still supported to load an HTML file as a view. However, Umbraco no longer supports AngularJS and the HTML file will be loaded into the DOM as-is. You will have to implement all the logic yourself.
You can use the following code as a starting point. This will give you a view looking like this, where the user can enter a code and click a button to verify the code. This is similar to the built-in view in Umbraco. In a real world scenario, you would probably want to authenticate the user in a different way.
The following code is an example of a custom 2FA login screen using Lit. This is the recommended way of creating a custom 2FA login screen. Lit is a light-weight library that augments the Custom Elements API to provide a declarative, performant, and interoperable way to create web components.
The element registers two properties: providers and returnPath. These properties are used to render the view. The providers property is an array of strings, where each string is the name of a 2FA provider. The returnPath is the path to redirect to after a successful login. Both supplied by the login screen automatically.
We need to register the custom view using a composer. This can be done on the IUmbracoBuilder in your startup or a composer. In this case, we will add a composer to your project. This composer will overwrite the IBackOfficeTwoFactorOptions to use the custom view.