diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor
index fc7619c2be..d8c9db1bb6 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor
@@ -88,8 +88,8 @@
IconName="@BitIconName.RecycleBin" />
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.Utils.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.Utils.cs
index 3e024b68b9..a218c0e5cf 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.Utils.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.Utils.cs
@@ -1,6 +1,7 @@
using System.Text;
using System.Diagnostics;
using System.Runtime.CompilerServices;
+using Boilerplate.Shared.Controllers.Identity;
//#if (signalR == true)
using Microsoft.AspNetCore.SignalR.Client;
//#endif
@@ -9,9 +10,10 @@ namespace Boilerplate.Client.Core.Components.Layout;
public partial class AppDiagnosticModal
{
- [AutoInject] Cookie cookie = default!;
- [AutoInject] AuthManager authManager = default!;
- [AutoInject] IStorageService storageService = default!;
+ [AutoInject] private Cookie cookie = default!;
+ [AutoInject] private AuthManager authManager = default!;
+ [AutoInject] private IStorageService storageService = default!;
+ [AutoInject] private IUserController userController = default!;
private static async Task ThrowTestException()
{
@@ -97,13 +99,13 @@ await Task.Run(() =>
SnackBarService.Show("Memory After GC", GetMemoryUsage());
}
- string GetMemoryUsage()
+ private string GetMemoryUsage()
{
long memory = Environment.WorkingSet;
return $"{memory / (1024.0 * 1024.0):F2} MB";
}
- async Task ClearCache()
+ private async Task ClearData()
{
try
{
@@ -118,6 +120,15 @@ async Task ClearCache()
await cookie.Remove(item.Name!);
}
+ if ((await AuthenticationStateTask).User.IsAuthenticated())
+ {
+ await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
+
+ // since the localStorage is used to store configured webauthn users and it is already cleared above, we can ignore the following line.
+ // we kept it commented for future refrences.
+ //await JSRuntime.RemoveWebAuthnConfigured();
+ }
+
if (AppPlatform.IsBlazorHybrid is false)
{
await JSRuntime.InvokeVoidAsync("BitBswup.forceRefresh"); // Clears cache storages and uninstalls service-worker.
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/PasswordlessSection.razor b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/PasswordlessSection.razor
index 918afd886a..d312eb4634 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/PasswordlessSection.razor
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/PasswordlessSection.razor
@@ -8,13 +8,13 @@
@if (isConfigured)
{
-
+
@Localizer[nameof(AppStrings.DisablePasswordless)]
}
else
{
-
+
@Localizer[nameof(AppStrings.EnablePasswordless)]
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/SettingsPage.razor.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/SettingsPage.razor.cs
index 6fb471140a..74634ba4ba 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/SettingsPage.razor.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Authorized/Settings/SettingsPage.razor.cs
@@ -34,7 +34,10 @@ protected override async Task OnInitAsync()
try
{
user = (await PrerenderStateService.GetValue(() => HttpClient.GetFromJsonAsync("api/User/GetCurrentUser", JsonSerializerOptions.GetTypeInfo(), CurrentCancellationToken)))!;
- showPasswordless = await JSRuntime.IsWebAuthnAvailable() && AppPlatform.IsBlazorHybrid is false;
+ if (InPrerenderSession is false)
+ {
+ showPasswordless = await JSRuntime.IsWebAuthnAvailable() && AppPlatform.IsBlazorHybrid is false;
+ }
}
finally
{
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Identity/SignIn/SignInPage.razor.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Identity/SignIn/SignInPage.razor.cs
index 62692817c0..d257bf26c0 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Identity/SignIn/SignInPage.razor.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Pages/Identity/SignIn/SignInPage.razor.cs
@@ -38,6 +38,7 @@ public partial class SignInPage
private SignInPanelTab currentSignInPanelTab;
private readonly SignInRequestDto model = new();
private AppDataAnnotationsValidator? validatorRef;
+ private AuthenticatorAssertionRawResponse? webAuthnAssertion;
private Action unsubscribeIdentityHeaderBackLinkClicked = default!;
@@ -83,6 +84,7 @@ protected override async Task OnInitAsync()
if (source == TfaPayload)
{
+ webAuthnAssertion = null;
requiresTwoFactor = false;
model.TwoFactorCode = null;
}
@@ -105,20 +107,29 @@ private async Task DoSignIn()
{
if (requiresTwoFactor && string.IsNullOrWhiteSpace(model.TwoFactorCode)) return;
- CleanModel();
+ if (webAuthnAssertion is null)
+ {
+ CleanModel();
- if (validatorRef?.EditContext.Validate() is false) return;
+ if (validatorRef?.EditContext.Validate() is false) return;
- model.DeviceInfo = telemetryContext.Platform;
+ model.DeviceInfo = telemetryContext.Platform;
- requiresTwoFactor = await AuthManager.SignIn(model, CurrentCancellationToken);
+ requiresTwoFactor = await AuthManager.SignIn(model, CurrentCancellationToken);
- if (requiresTwoFactor)
- {
- PubSubService.Publish(ClientPubSubMessages.UPDATE_IDENTITY_HEADER_BACK_LINK, TfaPayload);
+ if (requiresTwoFactor)
+ {
+ PubSubService.Publish(ClientPubSubMessages.UPDATE_IDENTITY_HEADER_BACK_LINK, TfaPayload);
+ }
+ else
+ {
+ NavigationManager.NavigateTo(ReturnUrlQueryString ?? Urls.HomePage, replace: true);
+ }
}
else
{
+ var response = await identityController.VerifyWebAuthAndSignIn(new() { ClientResponse = webAuthnAssertion, TfaCode = model.TwoFactorCode }, CurrentCancellationToken);
+ await AuthManager.StoreTokens(response!, model.RememberMe);
NavigationManager.NavigateTo(ReturnUrlQueryString ?? Urls.HomePage, replace: true);
}
}
@@ -160,10 +171,9 @@ private async Task HandleOnPasswordlessSignIn()
{
var options = await identityController.GetWebAuthnAssertionOptions(CurrentCancellationToken);
- AuthenticatorAssertionRawResponse assertion;
try
{
- assertion = await JSRuntime.VerifyWebAuthnCredential(options);
+ webAuthnAssertion = await JSRuntime.VerifyWebAuthnCredential(options);
}
catch (Exception ex)
{
@@ -172,9 +182,11 @@ private async Task HandleOnPasswordlessSignIn()
return;
}
- var response = await identityController.VerifyWebAuthAndSignIn(assertion, CurrentCancellationToken);
+ var response = await identityController.VerifyWebAuthAndSignIn(new() { ClientResponse = webAuthnAssertion }, CurrentCancellationToken);
+
+ requiresTwoFactor = response.RequiresTwoFactor;
- if (response.RequiresTwoFactor)
+ if (requiresTwoFactor)
{
PubSubService.Publish(ClientPubSubMessages.UPDATE_IDENTITY_HEADER_BACK_LINK, TfaPayload);
}
@@ -186,6 +198,7 @@ private async Task HandleOnPasswordlessSignIn()
}
catch (KnownException e)
{
+ webAuthnAssertion = null;
SnackBarService.Error(e.Message);
}
finally
@@ -242,9 +255,16 @@ private async Task HandleOnSendTfaToken()
{
try
{
- CleanModel();
+ if (webAuthnAssertion is null)
+ {
+ CleanModel();
- await identityController.SendTwoFactorToken(model, CurrentCancellationToken);
+ await identityController.SendTwoFactorToken(model, CurrentCancellationToken);
+ }
+ else
+ {
+ await identityController.VerifyWebAuthAndSendTwoFactorToken(webAuthnAssertion, CurrentCancellationToken);
+ }
SnackBarService.Success(Localizer[nameof(AppStrings.TfaTokenSentMessage)]);
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Extensions/IJSRuntimeWebAuthnExtensions.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Extensions/IJSRuntimeWebAuthnExtensions.cs
index 49be599d9f..18cda6f79f 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Extensions/IJSRuntimeWebAuthnExtensions.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Extensions/IJSRuntimeWebAuthnExtensions.cs
@@ -20,7 +20,7 @@ public static ValueTask IsWebAuthnConfigured(this IJSRuntime jsRuntime, st
return jsRuntime.InvokeAsync("WebAuthn.isConfigured", username);
}
- public static ValueTask RemoveWebAuthnConfigured(this IJSRuntime jsRuntime, string username)
+ public static ValueTask RemoveWebAuthnConfigured(this IJSRuntime jsRuntime, string? username = null)
{
return jsRuntime.InvokeVoidAsync("WebAuthn.removeConfigured", username);
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts
index d3bb2b27db..b24b327f50 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts
@@ -1,105 +1,167 @@
class WebAuthn {
- private static STORE_KEY = 'configured-webauthn';
+ private static STORE_KEY = 'bit-webauthn';
public static isAvailable() {
return !!window.PublicKeyCredential;
}
- public static storeConfigured(username: string) {
+ public static isConfigured(username: string | undefined) {
const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
- storedCredentials.push(username);
- localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(storedCredentials));
+ return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
}
- public static isConfigured(username: string | undefined) {
+ public static storeConfigured(username: string) {
const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
- return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
+ storedCredentials.push(username);
+ localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(storedCredentials));
}
public static removeConfigured(username: string) {
const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
- localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(storedCredentials.filter(c => c !== username)));
+ localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(!!username ? storedCredentials.filter(c => c !== username) : []));
}
public static async createCredential(options: PublicKeyCredentialCreationOptions) {
- if (typeof options.challenge === 'string') {
- options.challenge = WebAuthn.urlToArray(options.challenge);
- }
+ options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
+
+ options.user.id = WebAuthn.ToArrayBuffer(options.user.id, 'user.id');
+
+ options.excludeCredentials?.forEach(function (cred) {
+ cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
+ });
- if (typeof options.user.id === 'string') {
- options.user.id = WebAuthn.urlToArray(options.user.id);
+ if (options.authenticatorSelection?.authenticatorAttachment === null) {
+ options.authenticatorSelection.authenticatorAttachment = undefined;
}
if (options.rp.id === null) {
options.rp.id = undefined;
}
- for (let cred of options.excludeCredentials || []) {
- if (typeof cred.id !== 'string') continue;
-
- cred.id = WebAuthn.urlToArray(cred.id);
- }
-
const credential = await navigator.credentials.create({ publicKey: options }) as PublicKeyCredential;
+
const response = credential.response as AuthenticatorAttestationResponse;
- return {
- id: WebAuthn.base64ToUrl(credential.id),
- rawId: WebAuthn.arrayToUrl(credential.rawId),
+ // Move data into Arrays incase it is super long
+ const attestationObject = new Uint8Array(response.attestationObject);
+ const clientDataJSON = new Uint8Array(response.clientDataJSON);
+ const rawId = new Uint8Array(credential.rawId);
+
+ const result = {
+ id: credential.id,
+ rawId: WebAuthn.ToBase64Url(rawId),
type: credential.type,
clientExtensionResults: credential.getClientExtensionResults(),
response: {
- attestationObject: WebAuthn.arrayToUrl(response.attestationObject),
- clientDataJSON: WebAuthn.arrayToUrl(response.clientDataJSON),
+ attestationObject: WebAuthn.ToBase64Url(attestationObject),
+ clientDataJSON: WebAuthn.ToBase64Url(clientDataJSON),
transports: response.getTransports ? response.getTransports() : []
}
};
+
+ return result;
}
public static async verifyCredential(options: PublicKeyCredentialRequestOptions) {
- if (typeof options.challenge === 'string') {
- options.challenge = WebAuthn.urlToArray(options.challenge);
- }
+ options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
+
+ options.allowCredentials?.forEach(function (cred) {
+ cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
+ });
- if (options.allowCredentials) {
- for (let i = 0; i < options.allowCredentials.length; i++) {
- const id = options.allowCredentials[i].id;
- if (typeof id === 'string') {
- options.allowCredentials[i].id = WebAuthn.urlToArray(id);
- }
- }
- }
const credential = await navigator.credentials.get({ publicKey: options }) as PublicKeyCredential;
+
const response = credential.response as AuthenticatorAssertionResponse;
- return {
+ // Move data into Arrays incase it is super long
+ let authenticatorData = new Uint8Array(response.authenticatorData);
+ let clientDataJSON = new Uint8Array(response.clientDataJSON);
+ let rawId = new Uint8Array(credential.rawId);
+ let signature = new Uint8Array(response.signature);
+ let userHandle = new Uint8Array(response.userHandle || []);
+
+ var result = {
id: credential.id,
- rawId: WebAuthn.arrayToUrl(credential.rawId),
+ rawId: WebAuthn.ToBase64Url(rawId),
type: credential.type,
clientExtensionResults: credential.getClientExtensionResults(),
response: {
- authenticatorData: WebAuthn.arrayToUrl(response.authenticatorData),
- clientDataJSON: WebAuthn.arrayToUrl(response.clientDataJSON),
- userHandle: response.userHandle && response.userHandle.byteLength > 0 ? WebAuthn.arrayToUrl(response.userHandle) : undefined,
- signature: WebAuthn.arrayToUrl(response.signature)
+ authenticatorData: WebAuthn.ToBase64Url(authenticatorData),
+ clientDataJSON: WebAuthn.ToBase64Url(clientDataJSON),
+ userHandle: userHandle !== null ? WebAuthn.ToBase64Url(userHandle) : null,
+ signature: WebAuthn.ToBase64Url(signature)
}
}
+
+ return result;
}
- private static arrayToUrl(value: ArrayBuffer): string {
- return btoa(String.fromCharCode(...new Uint8Array(value))).replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
- }
+ private static ToBase64Url(value: any) {
+ // Array or ArrayBuffer to Uint8Array
+ if (Array.isArray(value)) {
+ value = Uint8Array.from(value);
+ }
- private static urlToArray(value: string): Uint8Array {
- return Uint8Array.from(atob(value.replace(/-/g, "+").replace(/_/g, "/")), c => c.charCodeAt(0));
- }
+ if (value instanceof ArrayBuffer) {
+ value = new Uint8Array(value);
+ }
- private static base64ToUrl(value: string): string {
- return value.replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
- }
+ // Uint8Array to base64
+ if (value instanceof Uint8Array) {
+ var str = "";
+ var len = value.byteLength;
+
+ for (var i = 0; i < len; i++) {
+ str += String.fromCharCode(value[i]);
+ }
+ value = window.btoa(str);
+ }
+
+ if (typeof value !== "string") {
+ throw new Error("could not coerce to string");
+ }
+
+ // base64 to base64url
+ // NOTE: "=" at the end of challenge is optional, strip it off here
+ value = value.replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
+
+ return value;
+ };
+
+ private static ToArrayBuffer(value: any, name: string) {
+ if (typeof value === "string") {
+ // base64url to base64
+ value = value.replace(/-/g, "+").replace(/_/g, "/");
+
+ // base64 to Uint8Array
+ var str = window.atob(value);
+ var bytes = new Uint8Array(str.length);
+ for (var i = 0; i < str.length; i++) {
+ bytes[i] = str.charCodeAt(i);
+ }
+ value = bytes;
+ }
+
+ // Array to Uint8Array
+ if (Array.isArray(value)) {
+ value = new Uint8Array(value);
+ }
+
+ // Uint8Array to ArrayBuffer
+ if (value instanceof Uint8Array) {
+ value = value.buffer;
+ }
+
+ // error if none of the above worked
+ if (!(value instanceof ArrayBuffer)) {
+ throw new TypeError("could not coerce '" + name + "' to ArrayBuffer");
+ }
+
+ return value;
+ };
}
(window as any).WebAuthn = WebAuthn;
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/wwwroot/images/os/unknown.png b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/wwwroot/images/os/unknown.png
new file mode 100644
index 0000000000..a440991e8f
Binary files /dev/null and b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/wwwroot/images/os/unknown.png differ
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Web/wwwroot/images/icons/bit-logo.png b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Web/wwwroot/images/icons/bit-logo.png
new file mode 100644
index 0000000000..1629cbe86b
Binary files /dev/null and b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Web/wwwroot/images/icons/bit-logo.png differ
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.WebAuthn.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.WebAuthn.cs
index ba63186b45..6f7aadb423 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.WebAuthn.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.WebAuthn.cs
@@ -22,8 +22,8 @@ public async Task GetWebAuthnAssertionOptions(CancellationToke
var extensions = new AuthenticationExtensionsClientInputs
{
+ Extensions = true,
UserVerificationMethod = true,
- Extensions = true
};
var options = fido2.GetAssertionOptions(new GetAssertionOptionsParams
@@ -34,7 +34,7 @@ public async Task GetWebAuthnAssertionOptions(CancellationToke
});
var key = new string([.. options.Challenge.Select(b => (char)b)]);
- await cache.SetAsync(key, Encoding.UTF8.GetBytes(options.ToJson()), cancellationToken);
+ await cache.SetAsync(key, Encoding.UTF8.GetBytes(options.ToJson()), new() { SlidingExpiration = TimeSpan.FromMinutes(3) }, cancellationToken);
return options;
}
@@ -48,22 +48,36 @@ public async Task VerifyWebAuthAssertion(AuthenticatorAss
}
[HttpPost, Produces()]
- public async Task VerifyWebAuthAndSignIn(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken)
+ public async Task VerifyWebAuthAndSignIn(VerifyWebAuthnAndSignInDto request, CancellationToken cancellationToken)
{
- var (verifyResult, credential) = await Verify(clientResponse, cancellationToken);
+ var (verifyResult, credential) = await Verify(request.ClientResponse, cancellationToken);
var user = await userManager.FindByIdAsync(credential.UserId.ToString())
?? throw new ResourceNotFoundException();
var (otp, _) = await GenerateAutomaticSignInLink(user, null, "WebAuthn");
- credential.SignCount = verifyResult.SignCount;
+ if (user.TwoFactorEnabled is false || request.TfaCode is not null)
+ {
+ credential.SignCount = verifyResult.SignCount;
+ DbContext.WebAuthnCredential.Update(credential);
+ await DbContext.SaveChangesAsync(cancellationToken);
+ }
- DbContext.WebAuthnCredential.Update(credential);
+ await SignIn(new() { Otp = otp, TwoFactorCode = request.TfaCode }, user, cancellationToken);
+ }
- await DbContext.SaveChangesAsync(cancellationToken);
+ [HttpPost]
+ public async Task VerifyWebAuthAndSendTwoFactorToken(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken)
+ {
+ var (verifyResult, credential) = await Verify(clientResponse, cancellationToken);
+
+ var user = await userManager.FindByIdAsync(credential.UserId.ToString())
+ ?? throw new ResourceNotFoundException();
+
+ var (otp, _) = await GenerateAutomaticSignInLink(user, null, "WebAuthn");
- await SignIn(new() { Otp = otp }, user, cancellationToken);
+ await SendTwoFactorToken(new() { Otp = otp }, user, cancellationToken);
}
@@ -79,7 +93,8 @@ public async Task VerifyWebAuthAndSignIn(AuthenticatorAssertionRawResponse clien
var jsonOptions = Encoding.UTF8.GetString(cachedBytes);
var options = AssertionOptions.FromJson(jsonOptions);
- await cache.RemoveAsync(key, cancellationToken);
+ // since the TFA needs this option we won't remove it from cache manually and just wait for it to expire.
+ // await cache.RemoveAsync(key, cancellationToken);
var credential = (await DbContext.WebAuthnCredential.FirstOrDefaultAsync(c => c.Id == clientResponse.Id, cancellationToken))
?? throw new ResourceNotFoundException();
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.cs
index 353d70da57..daf1a5cd63 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/IdentityController.cs
@@ -92,7 +92,7 @@ public async Task SignIn(SignInRequestDto request, CancellationToken cancellatio
{
request.PhoneNumber = phoneService.NormalizePhoneNumber(request.PhoneNumber);
- var user = await userManager.FindUserAsync(request)
+ var user = await userManager.FindUserAsync(request)
?? throw new UnauthorizedException(Localizer[nameof(AppStrings.InvalidUserCredentials)]).WithData("Identifier", request);
await SignIn(request, user, cancellationToken);
@@ -341,8 +341,15 @@ public async Task SendOtp(IdentityRequestDto request, string? returnUrl = null,
public async Task SendTwoFactorToken(SignInRequestDto request, CancellationToken cancellationToken)
{
request.PhoneNumber = phoneService.NormalizePhoneNumber(request.PhoneNumber);
- var user = await userManager.FindUserAsync(request) ?? throw new ResourceNotFoundException(Localizer[nameof(AppStrings.UserNotFound)]).WithData("Identifier", request);
+ var user = await userManager.FindUserAsync(request)
+ ?? throw new ResourceNotFoundException(Localizer[nameof(AppStrings.UserNotFound)]).WithData("Identifier", request);
+
+ await SendTwoFactorToken(request, user, cancellationToken);
+ }
+
+ private async Task SendTwoFactorToken(SignInRequestDto request, User user, CancellationToken cancellationToken)
+ {
if (user.TwoFactorEnabled is false)
throw new BadRequestException().WithData("UserId", user.Id);
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/UserController.WebAuthn.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/UserController.WebAuthn.cs
index 51230b855a..910c1db097 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/UserController.WebAuthn.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Controllers/Identity/UserController.WebAuthn.cs
@@ -3,7 +3,6 @@
using Microsoft.Extensions.Caching.Distributed;
using Fido2NetLib;
using Fido2NetLib.Objects;
-using Boilerplate.Shared.Dtos.Identity;
using Boilerplate.Server.Api.Models.Identity;
namespace Boilerplate.Server.Api.Controllers.Identity;
@@ -32,22 +31,35 @@ public async Task GetWebAuthnCredentialOptions(Cancella
DisplayName = user.DisplayName,
};
+ var authenticatorSelection = new AuthenticatorSelection
+ {
+ ResidentKey = ResidentKeyRequirement.Required,
+ UserVerification = UserVerificationRequirement.Preferred
+ };
+
+ //var authenticatorSelection = new AuthenticatorSelection
+ //{
+ // AuthenticatorAttachment = AuthenticatorAttachment.Platform
+ //};
+
+ var extensions = new AuthenticationExtensionsClientInputs
+ {
+ CredProps = true,
+ Extensions = true,
+ UserVerificationMethod = true,
+ };
+
var options = fido2.RequestNewCredential(new RequestNewCredentialParams
{
User = fidoUser,
- ExcludeCredentials = [.. existingKeys],
- AuthenticatorSelection = AuthenticatorSelection.Default,
+ ExcludeCredentials = [], //[.. existingKeys],
+ AuthenticatorSelection = authenticatorSelection,
AttestationPreference = AttestationConveyancePreference.None,
- Extensions = new AuthenticationExtensionsClientInputs
- {
- CredProps = true,
- Extensions = true,
- UserVerificationMethod = true,
- }
+ Extensions = extensions
});
var key = GetWebAuthnCacheKey(userId);
- await cache.SetAsync(key, Encoding.UTF8.GetBytes(options.ToJson()), cancellationToken);
+ await cache.SetAsync(key, Encoding.UTF8.GetBytes(options.ToJson()), new() { SlidingExpiration = TimeSpan.FromMinutes(3) }, cancellationToken);
return options;
}
@@ -114,6 +126,22 @@ public async Task DeleteWebAuthnCredential(byte[] credentialId, CancellationToke
await DbContext.SaveChangesAsync(cancellationToken);
}
+ [HttpDelete]
+ public async Task DeleteAllWebAuthnCredentials(CancellationToken cancellationToken)
+ {
+ var userId = User.GetUserId();
+ var user = await userManager.FindByIdAsync(userId.ToString())
+ ?? throw new ResourceNotFoundException();
+
+ var entities = await DbContext.WebAuthnCredential.Where(c => c.UserId == userId).ToListAsync(cancellationToken);
+
+ if (entities is null || entities.Count == 0) return;
+
+ DbContext.WebAuthnCredential.RemoveRange(entities);
+
+ await DbContext.SaveChangesAsync(cancellationToken);
+ }
+
private static string GetWebAuthnCacheKey(Guid userId) => $"WebAuthn_Options_{userId}";
private async Task IsCredentialIdUniqueToUser(IsCredentialIdUniqueToUserParams args, CancellationToken cancellationToken)
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Models/Identity/WebAuthnCredential.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Models/Identity/WebAuthnCredential.cs
index a44563d2cc..1846cea1a6 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Models/Identity/WebAuthnCredential.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Models/Identity/WebAuthnCredential.cs
@@ -2,6 +2,11 @@
namespace Boilerplate.Server.Api.Models.Identity;
+///
+/// This model is used by the Fido2 lib to store and retrieve the data of the browser credential api for `Web Authentication`.
+///
+/// More info:
+///
public class WebAuthnCredential
{
public required byte[] Id { get; set; }
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Program.Services.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Program.Services.cs
index b6cdbdf3f8..207fa1b314 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Program.Services.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/Program.Services.cs
@@ -10,6 +10,7 @@
using Microsoft.AspNetCore.ResponseCompression;
using System.Security.Cryptography.X509Certificates;
using Twilio;
+using Fido2NetLib;
using PhoneNumbers;
using FluentStorage;
using FluentStorage.Blobs;
@@ -22,7 +23,6 @@
using Boilerplate.Server.Api.Controllers;
using Boilerplate.Server.Api.Models.Identity;
using Boilerplate.Server.Api.Services.Identity;
-
namespace Boilerplate.Server.Api;
public static partial class Program
@@ -301,20 +301,24 @@ void AddDbContext(DbContextOptionsBuilder options)
services.AddFido2(options =>
{
- var trustedOrigins = appSettings.AllowedOrigins
- .Union([
- //#if (api == "Integrated")
- new Uri("http://localhost:5030/")
- //#else
- , new Uri("http://localhost:5031/")
- //#endif
- ]);
-
- options.TimestampDriftTolerance = 1000;
- options.ServerName = "Boilerplate WebAuthn";
- options.ServerDomain = trustedOrigins.First().Host;
- options.Origins = new HashSet(trustedOrigins.Select(uri => uri.AbsoluteUri));
- options.ServerIcon = "";
+
+ });
+
+ services.AddScoped(sp =>
+ {
+ var webAppUrl = sp.GetRequiredService()
+ .HttpContext!.Request.GetWebAppUrl();
+
+ var options = new Fido2Configuration
+ {
+ TimestampDriftTolerance = 1000,
+ ServerName = "Boilerplate WebAuthn",
+ ServerDomain = webAppUrl.Host,
+ Origins = new HashSet([webAppUrl.AbsoluteUri]),
+ ServerIcon = new Uri(webAppUrl, "images/icons/bit-logo.png").ToString()
+ };
+
+ return options;
});
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/ServerApiSettings.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/ServerApiSettings.cs
index 306157b6c0..606a98beef 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/ServerApiSettings.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/ServerApiSettings.cs
@@ -48,9 +48,9 @@ public partial class ServerApiSettings : SharedSettings
public ResponseCachingOptions ResponseCaching { get; set; } = default!;
///
- /// Defines the list of origins permitted for CORS access to the API. These origins are `also` valid for use as return URLs after social sign-ins and for generating URLs in emails.
+ /// Lists the permitted origins for CORS requests, return URLs following social sign-in and email confirmation, etc., along with allowed origins for Web Auth.
///
- public Uri[] AllowedOrigins { get; set; } = [];
+ public Uri[] TrustedOrigins { get; set; } = [];
//#if (module == "Admin" || module == "Sales")
[Required]
@@ -113,8 +113,8 @@ public override IEnumerable Validate(ValidationContext validat
internal bool IsAllowedOrigin(Uri origin)
{
- return AllowedOrigins.Any(allowedOrigin => allowedOrigin == origin)
- || AllowedOriginsRegex().IsMatch(origin.ToString());
+ return TrustedOrigins.Any(trustedOrigin => trustedOrigin == origin)
+ || TrustedOriginsRegex().IsMatch(origin.ToString());
}
//-:cnd:noEmit
@@ -127,7 +127,7 @@ internal bool IsAllowedOrigin(Uri origin)
[GeneratedRegex(@"^(http|https|app):\/\/(localhost|0\.0\.0\.0|0\.0\.0\.1|127\.0\.0\.1)(:\d+)?(\/.*)?$")]
#endif
//+:cnd:noEmit
- private partial Regex AllowedOriginsRegex();
+ private partial Regex TrustedOriginsRegex();
}
public partial class AppIdentityOptions : IdentityOptions
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/appsettings.json b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/appsettings.json
index c0ce66bac6..225695696c 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/appsettings.json
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Server/Boilerplate.Server.Api/appsettings.json
@@ -131,8 +131,8 @@
}
},
"AllowedHosts": "*",
- "AllowedOrigins": [],
- "AllowedOrigins_Comment": "Defines the list of origins permitted for CORS access to the API. These origins are also valid for use as return URLs after social sign-ins and for generating URLs in emails.",
+ "TrustedOrigins": [],
+ "TrustedOrigins_Comment": "Lists the permitted origins for CORS requests, return URLs following social sign-in and email confirmation, etc., along with allowed origins for Web Auth.",
"ForwardedHeaders": {
"ForwardedHeaders": "All",
"ForwardedHeaders_Comment": "These values apply only if your backend is hosted behind a CDN (such as `Cloudflare`).",
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IIdentityController.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IIdentityController.cs
index 81ac52e4c2..aa2b7b44b0 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IIdentityController.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IIdentityController.cs
@@ -55,5 +55,8 @@ public interface IIdentityController : IAppController
Task VerifyWebAuthAssertion(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken);
[HttpPost]
- Task VerifyWebAuthAndSignIn(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken) => default!;
+ Task VerifyWebAuthAndSignIn(VerifyWebAuthnAndSignInDto request, CancellationToken cancellationToken) => default!;
+
+ [HttpPost]
+ Task VerifyWebAuthAndSendTwoFactorToken(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken);
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IUserController.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IUserController.cs
index 720a75d8b6..9ecca46c8b 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IUserController.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Controllers/Identity/IUserController.cs
@@ -57,4 +57,7 @@ public interface IUserController : IAppController
[HttpDelete]
Task DeleteWebAuthnCredential(byte[] credentialId, CancellationToken cancellationToken);
+
+ [HttpDelete]
+ Task DeleteAllWebAuthnCredentials(CancellationToken cancellationToken);
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/AppJsonContext.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/AppJsonContext.cs
index 43783c773a..c28ab09a1e 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/AppJsonContext.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/AppJsonContext.cs
@@ -14,6 +14,7 @@
//#if (notification == true)
using Boilerplate.Shared.Dtos.PushNotification;
//#endif
+using Boilerplate.Shared.Dtos.Identity;
using Boilerplate.Shared.Dtos.Statistics;
namespace Boilerplate.Shared.Dtos;
@@ -55,6 +56,7 @@ namespace Boilerplate.Shared.Dtos;
[JsonSerializable(typeof(AuthenticatorAttestationRawResponse))]
[JsonSerializable(typeof(CredentialCreateOptions))]
[JsonSerializable(typeof(VerifyAssertionResult))]
+[JsonSerializable(typeof(VerifyWebAuthnAndSignInDto))]
public partial class AppJsonContext : JsonSerializerContext
{
}
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/Identity/VerifyWebAuthnAndSignInDto.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/Identity/VerifyWebAuthnAndSignInDto.cs
new file mode 100644
index 0000000000..65508939d0
--- /dev/null
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/Identity/VerifyWebAuthnAndSignInDto.cs
@@ -0,0 +1,10 @@
+using Fido2NetLib;
+
+namespace Boilerplate.Shared.Dtos.Identity;
+
+public partial class VerifyWebAuthnAndSignInDto
+{
+ public required AuthenticatorAssertionRawResponse ClientResponse { get; set; }
+
+ public string? TfaCode { get; set; }
+}