From d5e033b5daea7d2c3abbb5e68f80967ba781dd7d Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Tue, 11 Mar 2025 17:41:33 +0330
Subject: [PATCH 01/20] fix WebAuthn issues in Boilerplate #10228

---
 .../Layout/AppDiagnosticModal.razor           |  6 ++++
 .../Layout/AppDiagnosticModal.razor.cs        |  9 +++++
 .../Settings/PasswordlessSection.razor        |  4 +--
 .../Authorized/Settings/SettingsPage.razor.cs |  5 ++-
 .../Scripts/WebAuthn.ts                       | 35 ++++++++++---------
 .../Identity/UserController.WebAuthn.cs       | 31 +++++++++++-----
 .../Models/Identity/WebAuthnCredential.cs     |  5 +++
 .../Controllers/Identity/IUserController.cs   |  3 ++
 8 files changed, 70 insertions(+), 28 deletions(-)

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..1569c32697 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
@@ -145,6 +145,12 @@
                            Variant="BitVariant.Text"
                            OnClick="() => isLogModalOpen = false"
                            IconName="@BitIconName.ChromeClose" />
+                <BitButton IconOnly
+                           AutoLoading
+                           Color="BitColor.Info"
+                           Variant="BitVariant.Text"
+                           IconName="@BitIconName.Delete"
+                           OnClick="DeleteAllWebAuthnCredentials" />
             </BitStack>
             <br />
             <BitText Class="log-modal" Color="GetColor(selectedLog?.Level)">
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
index 38944c9096..3b19c69c9e 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
@@ -3,6 +3,7 @@
 //#endif
 using Boilerplate.Shared.Controllers.Diagnostics;
 using Boilerplate.Client.Core.Services.DiagnosticLog;
+using Boilerplate.Shared.Controllers.Identity;
 
 namespace Boilerplate.Client.Core.Components.Layout;
 
@@ -41,6 +42,7 @@ public partial class AppDiagnosticModal
     //#if (notification == true)
     [AutoInject] private IPushNotificationService pushNotificationService = default!;
     //#endif
+    [AutoInject] private IUserController userController = default!;
 
     protected override Task OnInitAsync()
     {
@@ -154,6 +156,13 @@ private void ResetLogs()
         FilterLogs();
     }
 
+    private async Task DeleteAllWebAuthnCredentials()
+    {
+        if ((await AuthenticationStateTask).User.IsAuthenticated() is false) return;
+
+        await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
+    }
+
     private static BitColor GetColor(LogLevel? level)
     {
         return level switch
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 @@
         <br />
         @if (isConfigured)
         {
-            <BitButton OnClick="WrapHandled(DisablePasswordless)" Variant="BitVariant.Outline" Color="BitColor.Warning">
+            <BitButton AutoLoading OnClick="WrapHandled(DisablePasswordless)" Variant="BitVariant.Outline" Color="BitColor.Warning">
                 @Localizer[nameof(AppStrings.DisablePasswordless)]
             </BitButton>
         }
         else
         {
-            <BitButton OnClick="WrapHandled(EnablePasswordless)">
+            <BitButton AutoLoading OnClick="WrapHandled(EnablePasswordless)">
                 @Localizer[nameof(AppStrings.EnablePasswordless)]
             </BitButton>
         }
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<UserDto>(), 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/Scripts/WebAuthn.ts b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts
index d3bb2b27db..db4166174b 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
@@ -23,12 +23,13 @@ class WebAuthn {
 
 
     public static async createCredential(options: PublicKeyCredentialCreationOptions) {
+        console.log(options)
         if (typeof options.challenge === 'string') {
-            options.challenge = WebAuthn.urlToArray(options.challenge);
+            options.challenge = WebAuthn.stringToBinary(options.challenge);
         }
 
         if (typeof options.user.id === 'string') {
-            options.user.id = WebAuthn.urlToArray(options.user.id);
+            options.user.id = WebAuthn.stringToBinary(options.user.id);
         }
 
         if (options.rp.id === null) {
@@ -38,20 +39,20 @@ class WebAuthn {
         for (let cred of options.excludeCredentials || []) {
             if (typeof cred.id !== 'string') continue;
 
-            cred.id = WebAuthn.urlToArray(cred.id);
+            cred.id = WebAuthn.stringToBinary(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),
+            id: WebAuthn.base64ToString(credential.id),
+            rawId: WebAuthn.binaryToString(credential.rawId),
             type: credential.type,
             clientExtensionResults: credential.getClientExtensionResults(),
             response: {
-                attestationObject: WebAuthn.arrayToUrl(response.attestationObject),
-                clientDataJSON: WebAuthn.arrayToUrl(response.clientDataJSON),
+                attestationObject: WebAuthn.binaryToString(response.attestationObject),
+                clientDataJSON: WebAuthn.binaryToString(response.clientDataJSON),
                 transports: response.getTransports ? response.getTransports() : []
             }
         };
@@ -59,14 +60,14 @@ class WebAuthn {
 
     public static async verifyCredential(options: PublicKeyCredentialRequestOptions) {
         if (typeof options.challenge === 'string') {
-            options.challenge = WebAuthn.urlToArray(options.challenge);
+            options.challenge = WebAuthn.stringToBinary(options.challenge);
         }
 
         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);
+                    options.allowCredentials[i].id = WebAuthn.stringToBinary(id);
                 }
             }
         }
@@ -75,29 +76,29 @@ class WebAuthn {
 
         return {
             id: credential.id,
-            rawId: WebAuthn.arrayToUrl(credential.rawId),
+            rawId: WebAuthn.binaryToString(credential.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.binaryToString(response.authenticatorData),
+                clientDataJSON: WebAuthn.binaryToString(response.clientDataJSON),
+                userHandle: response.userHandle && response.userHandle.byteLength > 0 ? WebAuthn.binaryToString(response.userHandle) : undefined,
+                signature: WebAuthn.binaryToString(response.signature)
             }
         }
     }
 
 
 
-    private static arrayToUrl(value: ArrayBuffer): string {
+    private static binaryToString(value: ArrayBuffer): string {
         return btoa(String.fromCharCode(...new Uint8Array(value))).replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
     }
 
-    private static urlToArray(value: string): Uint8Array {
+    private static stringToBinary(value: string): Uint8Array {
         return Uint8Array.from(atob(value.replace(/-/g, "+").replace(/_/g, "/")), c => c.charCodeAt(0));
     }
 
-    private static base64ToUrl(value: string): string {
+    private static base64ToString(value: string): string {
         return value.replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
     }
 }
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..beda9be5ba 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;
@@ -36,14 +35,14 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
         {
             User = fidoUser,
             ExcludeCredentials = [.. existingKeys],
-            AuthenticatorSelection = AuthenticatorSelection.Default,
+            AuthenticatorSelection = new() { AuthenticatorAttachment = AuthenticatorAttachment.Platform },
             AttestationPreference = AttestationConveyancePreference.None,
-            Extensions = new AuthenticationExtensionsClientInputs
-            {
-                CredProps = true,
-                Extensions = true,
-                UserVerificationMethod = true,
-            }
+            //Extensions = new AuthenticationExtensionsClientInputs
+            //{
+            //    CredProps = true,
+            //    Extensions = true,
+            //    UserVerificationMethod = true,
+            //}
         });
 
         var key = GetWebAuthnCacheKey(userId);
@@ -114,6 +113,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<bool> 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;
 
+/// <summary>
+/// This model is used by the Fido2 lib to store and retrieve the data of the browser credential api for `Web Authentication`.
+/// <br />
+/// More info: <see href="https://github.com/passwordless-lib/fido2-net-lib"/>
+/// </summary>
 public class WebAuthnCredential
 {
     public required byte[] Id { get; set; }
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);
 }

From c9a58dbc6cd90916602106e4977fc518ac567088 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Tue, 11 Mar 2025 17:54:12 +0330
Subject: [PATCH 02/20] fix diag modal

---
 .../Layout/AppDiagnosticModal.razor           | 12 +++++------
 .../Layout/AppDiagnosticModal.razor.Utils.cs  | 21 ++++++++++++++-----
 .../Layout/AppDiagnosticModal.razor.cs        | 10 +--------
 .../IJSRuntimeWebAuthnExtensions.cs           |  2 +-
 .../Scripts/WebAuthn.ts                       |  3 +--
 5 files changed, 25 insertions(+), 23 deletions(-)

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 1569c32697..43cd6d8504 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
@@ -92,6 +92,12 @@
                            Title="Clear cache"
                            Color="BitColor.SecondaryBackground"
                            IconName="@BitIconName.Clear" />
+                <BitButton IconOnly AutoLoading
+                           Color="BitColor.Info"
+                           Variant="BitVariant.Text"
+                           IconName="@BitIconName.Delete"
+                           Title="Clear WebAuthn"
+                           OnClick="ClearWebAuthn" />
             </BitStack>
 
             <BitBasicList @ref="logStackRef"
@@ -145,12 +151,6 @@
                            Variant="BitVariant.Text"
                            OnClick="() => isLogModalOpen = false"
                            IconName="@BitIconName.ChromeClose" />
-                <BitButton IconOnly
-                           AutoLoading
-                           Color="BitColor.Info"
-                           Variant="BitVariant.Text"
-                           IconName="@BitIconName.Delete"
-                           OnClick="DeleteAllWebAuthnCredentials" />
             </BitStack>
             <br />
             <BitText Class="log-modal" Color="GetColor(selectedLog?.Level)">
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..384590bfe3 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 ClearCache()
     {
         try
         {
@@ -127,4 +129,13 @@ async Task ClearCache()
             NavigationManager.Refresh(forceReload: true);
         }
     }
+
+    private async Task ClearWebAuthn()
+    {
+        if ((await AuthenticationStateTask).User.IsAuthenticated() is false) return;
+
+        await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
+
+        await JSRuntime.RemoveWebAuthnConfigured();
+    }
 }
diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
index 3b19c69c9e..d87d928e20 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
@@ -1,9 +1,9 @@
 //#if (signalR == true)
 using Microsoft.AspNetCore.SignalR.Client;
 //#endif
+using Boilerplate.Shared.Controllers.Identity;
 using Boilerplate.Shared.Controllers.Diagnostics;
 using Boilerplate.Client.Core.Services.DiagnosticLog;
-using Boilerplate.Shared.Controllers.Identity;
 
 namespace Boilerplate.Client.Core.Components.Layout;
 
@@ -42,7 +42,6 @@ public partial class AppDiagnosticModal
     //#if (notification == true)
     [AutoInject] private IPushNotificationService pushNotificationService = default!;
     //#endif
-    [AutoInject] private IUserController userController = default!;
 
     protected override Task OnInitAsync()
     {
@@ -156,13 +155,6 @@ private void ResetLogs()
         FilterLogs();
     }
 
-    private async Task DeleteAllWebAuthnCredentials()
-    {
-        if ((await AuthenticationStateTask).User.IsAuthenticated() is false) return;
-
-        await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
-    }
-
     private static BitColor GetColor(LogLevel? level)
     {
         return level switch
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<bool> IsWebAuthnConfigured(this IJSRuntime jsRuntime, st
         return jsRuntime.InvokeAsync<bool>("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 db4166174b..fbd08dfb62 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
@@ -18,12 +18,11 @@ class WebAuthn {
 
     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) {
-        console.log(options)
         if (typeof options.challenge === 'string') {
             options.challenge = WebAuthn.stringToBinary(options.challenge);
         }

From 33d86efb652bc47e38ddb820d9eaf286d7a0caa1 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Tue, 11 Mar 2025 17:58:56 +0330
Subject: [PATCH 03/20] add missing unknown.png

---
 .../wwwroot/images/os/unknown.png                 | Bin 0 -> 909 bytes
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/wwwroot/images/os/unknown.png

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 0000000000000000000000000000000000000000..a440991e8f089ceabd90b0d486f151c1c9192ec0
GIT binary patch
literal 909
zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz#^NA%Cx&(BWL^R}Ea{HEjtmSN
z`?>!lvI6;>1s;*b3=DcoAk4Vx(cMWvL5ULAh?3y^w370~qEv=}#LT=BJwMkF1yemk
zJrkk(%bI~|wxvdRrg?g5F>nAmtPE0&tPG4mmKP99L)jqLXfQH^#hHL?Lq;YB0U#X(
z#F_0ZVDT&<8wBo8W?^{2jG)mhWdKS|U}s<fsx&Y%Heg%;F%@JZ>jH>LQ-Ev`U;>)U
z1XdYjX#r$Gbr~8MfMnOmdVbt7+gc2$&c@TlF(ktM>~!xFevT3i$77QeH!4h0DZ1H_
z@^IgViLIM92f27@N^?u!ys@&r`OTUX)su5Qk`k2zgTlm4H7RZXVDo&=#qY)Y-k+H{
zxqtb*dcD$fd(zYP&HT>)f0kkBP8Rb=+ZTQPF4@2RW8}rAzdw>sSfxq4mY29T{d-W4
z<n#pod0ML<irqWX>AG5O{{!th{V2sA=jEpR*7xb}kTe%vEAZ{n%S};i0?VImkmWpc
zeCrOb^cxc!-uFGWc(z{kMXFQiw`9qC2gNJo<sWMAY2Ypts6Uex%g{WdcY2|%)l>No
z#(qa-UtBtSFy$6ofxz==R}0w09|rBoVn1`ZHr2Fu8{-k)FLM14vTHhz8*bX8H914L
z!TOIt-tpWYV#=3S1>Flx70P8Qcz*A2cg6Os(@ZvjVch4!fnw7ilwW(uTq1uWa8F3s
z<rhK+N^IIUUx+`fXVbP8$n)yi&$0hbLeyndhW>}ge+aC<J<)kO!ye-~t)@SY^vUdE
zd61bQU|z6Q*0O~2z*qCmk8Phj&Moq0$p4^e^>=T3koGI45A*yVN>v%JbKqYga)#3_
zaQcK9t1~Ze)%j>5Go{1jkHK`GrTI=qI_|0}_ud{}CSmM%H|b8p%9AUzcS`T&K63h{
z4&N!|>cWD}tDd`RPo6BX{Bo6;@9cD;yrU;yI3&H~kvX7S*3f>D!;1Oi+-dVaaLwA5
zYAAfR?$PvZSJP|V<AeAXy?L?l<IO1ybN<Ch&Ck8Xye~oB@8FO40Ij+^?k29q`hWUc
aJ_zp%Tah`}uI3^rO?kTdxvX<aXaWHJmR70&

literal 0
HcmV?d00001


From 389c4a38a9a8fe096a2dfd50fff1e82b1387ac90 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Tue, 11 Mar 2025 18:07:11 +0330
Subject: [PATCH 04/20] remove more options props

---
 .../Controllers/Identity/IdentityController.WebAuthn.cs       | 4 ++--
 .../Controllers/Identity/UserController.WebAuthn.cs           | 4 ++--
 2 files changed, 4 insertions(+), 4 deletions(-)

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..0bebe8f350 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
@@ -28,8 +28,8 @@ public async Task<AssertionOptions> GetWebAuthnAssertionOptions(CancellationToke
 
         var options = fido2.GetAssertionOptions(new GetAssertionOptionsParams
         {
-            Extensions = extensions,
-            AllowedCredentials = existingKeys,
+            //Extensions = extensions,
+            //AllowedCredentials = existingKeys,
             UserVerification = UserVerificationRequirement.Discouraged,
         });
 
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 beda9be5ba..365faa34be 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
@@ -34,9 +34,9 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
         var options = fido2.RequestNewCredential(new RequestNewCredentialParams
         {
             User = fidoUser,
-            ExcludeCredentials = [.. existingKeys],
+            //ExcludeCredentials = [.. existingKeys],
             AuthenticatorSelection = new() { AuthenticatorAttachment = AuthenticatorAttachment.Platform },
-            AttestationPreference = AttestationConveyancePreference.None,
+            //AttestationPreference = AttestationConveyancePreference.None,
             //Extensions = new AuthenticationExtensionsClientInputs
             //{
             //    CredProps = true,

From 5e7b2b596e9817f53051d994995e382f2da88799 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Tue, 11 Mar 2025 19:27:05 +0330
Subject: [PATCH 05/20] add logs

---
 .../Scripts/WebAuthn.ts                        | 14 ++++++++++----
 .../Identity/IdentityController.WebAuthn.cs    |  6 +++---
 .../Identity/UserController.WebAuthn.cs        | 18 +++++++++---------
 3 files changed, 22 insertions(+), 16 deletions(-)

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 fbd08dfb62..85ecf94ed4 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
@@ -23,6 +23,7 @@ class WebAuthn {
 
 
     public static async createCredential(options: PublicKeyCredentialCreationOptions) {
+        console.log(options)
         if (typeof options.challenge === 'string') {
             options.challenge = WebAuthn.stringToBinary(options.challenge);
         }
@@ -43,8 +44,8 @@ class WebAuthn {
 
         const credential = await navigator.credentials.create({ publicKey: options }) as PublicKeyCredential;
         const response = credential.response as AuthenticatorAttestationResponse;
-
-        return {
+        console.log('response:', response)
+        const result = {
             id: WebAuthn.base64ToString(credential.id),
             rawId: WebAuthn.binaryToString(credential.rawId),
             type: credential.type,
@@ -55,9 +56,12 @@ class WebAuthn {
                 transports: response.getTransports ? response.getTransports() : []
             }
         };
+        console.log('result:', result)
+        return result;
     }
 
     public static async verifyCredential(options: PublicKeyCredentialRequestOptions) {
+        console.log(options)
         if (typeof options.challenge === 'string') {
             options.challenge = WebAuthn.stringToBinary(options.challenge);
         }
@@ -72,8 +76,8 @@ class WebAuthn {
         }
         const credential = await navigator.credentials.get({ publicKey: options }) as PublicKeyCredential;
         const response = credential.response as AuthenticatorAssertionResponse;
-
-        return {
+        console.log('response:', response)
+        var result = {
             id: credential.id,
             rawId: WebAuthn.binaryToString(credential.rawId),
             type: credential.type,
@@ -85,6 +89,8 @@ class WebAuthn {
                 signature: WebAuthn.binaryToString(response.signature)
             }
         }
+        console.log('result:', result)
+        return result;
     }
 
 
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 0bebe8f350..f677138a13 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,14 +22,14 @@ public async Task<AssertionOptions> GetWebAuthnAssertionOptions(CancellationToke
 
         var extensions = new AuthenticationExtensionsClientInputs
         {
+            Extensions = true,
             UserVerificationMethod = true,
-            Extensions = true
         };
 
         var options = fido2.GetAssertionOptions(new GetAssertionOptionsParams
         {
-            //Extensions = extensions,
-            //AllowedCredentials = existingKeys,
+            Extensions = extensions,
+            AllowedCredentials = existingKeys,
             UserVerification = UserVerificationRequirement.Discouraged,
         });
 
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 365faa34be..2fac870fe1 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
@@ -34,15 +34,15 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
         var options = fido2.RequestNewCredential(new RequestNewCredentialParams
         {
             User = fidoUser,
-            //ExcludeCredentials = [.. existingKeys],
-            AuthenticatorSelection = new() { AuthenticatorAttachment = AuthenticatorAttachment.Platform },
-            //AttestationPreference = AttestationConveyancePreference.None,
-            //Extensions = new AuthenticationExtensionsClientInputs
-            //{
-            //    CredProps = true,
-            //    Extensions = true,
-            //    UserVerificationMethod = true,
-            //}
+            ExcludeCredentials = [.. existingKeys],
+            AuthenticatorSelection = AuthenticatorSelection.Default, // new() { AuthenticatorAttachment = AuthenticatorAttachment.Platform },
+            AttestationPreference = AttestationConveyancePreference.None,
+            Extensions = new AuthenticationExtensionsClientInputs
+            {
+                CredProps = true,
+                Extensions = true,
+                UserVerificationMethod = true,
+            }
         });
 
         var key = GetWebAuthnCacheKey(userId);

From b1a93f84e8e5d1dc630bbffed741cf8b8b504eb7 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 10:37:34 +0330
Subject: [PATCH 06/20] fix ts

---
 .../Scripts/WebAuthn.ts                       | 195 ++++++++++++++----
 .../Identity/UserController.WebAuthn.cs       |  22 +-
 2 files changed, 166 insertions(+), 51 deletions(-)

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 85ecf94ed4..03b3db6307 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
@@ -23,89 +23,196 @@ class WebAuthn {
 
 
     public static async createCredential(options: PublicKeyCredentialCreationOptions) {
-        console.log(options)
-        if (typeof options.challenge === 'string') {
-            options.challenge = WebAuthn.stringToBinary(options.challenge);
+        console.log(options);
+
+        //if (typeof options.challenge === 'string') {
+        //    options.challenge = WebAuthn.stringToBinary(options.challenge);
+        //}
+
+        options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
+
+        //if (typeof options.user.id === 'string') {
+        //    options.user.id = WebAuthn.stringToBinary(options.user.id);
+        //}
+
+        options.user.id = WebAuthn.ToArrayBuffer(options.user.id, 'user.id');
+
+        for (let cred of options.excludeCredentials || []) {
+            //if (typeof cred.id !== 'string') continue;
+            //cred.id = WebAuthn.stringToBinary(cred.id);
+            cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
         }
 
-        if (typeof options.user.id === 'string') {
-            options.user.id = WebAuthn.stringToBinary(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.stringToBinary(cred.id);
-        }
+        console.log('corrected:', options);
 
         const credential = await navigator.credentials.create({ publicKey: options }) as PublicKeyCredential;
+
+        console.log('credential:', credential);
+
         const response = credential.response as AuthenticatorAttestationResponse;
-        console.log('response:', response)
+
+        // 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: WebAuthn.base64ToString(credential.id),
-            rawId: WebAuthn.binaryToString(credential.rawId),
+            id: credential.id,
+            rawId: WebAuthn.ToBase64Url(rawId),
             type: credential.type,
             clientExtensionResults: credential.getClientExtensionResults(),
             response: {
-                attestationObject: WebAuthn.binaryToString(response.attestationObject),
-                clientDataJSON: WebAuthn.binaryToString(response.clientDataJSON),
+                attestationObject: WebAuthn.ToBase64Url(attestationObject),
+                clientDataJSON: WebAuthn.ToBase64Url(clientDataJSON),
                 transports: response.getTransports ? response.getTransports() : []
             }
         };
-        console.log('result:', result)
+
+        console.log('result:', result);
+
         return result;
     }
 
     public static async verifyCredential(options: PublicKeyCredentialRequestOptions) {
-        console.log(options)
-        if (typeof options.challenge === 'string') {
-            options.challenge = WebAuthn.stringToBinary(options.challenge);
-        }
+        console.log(options);
+
+        //if (typeof options.challenge === 'string') {
+        //    options.challenge = WebAuthn.stringToBinary(options.challenge);
+        //}
+
+        options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
+
+        //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.stringToBinary(id);
+        //        }
+        //    }
+        //}
+
+        options.allowCredentials?.forEach(function (cred) {
+            cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
+        });
+
+        console.log('corrected:', options);
 
-        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.stringToBinary(id);
-                }
-            }
-        }
         const credential = await navigator.credentials.get({ publicKey: options }) as PublicKeyCredential;
+
+        console.log('credential:', credential);
+
         const response = credential.response as AuthenticatorAssertionResponse;
-        console.log('response:', response)
+
+        // 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.binaryToString(credential.rawId),
+            rawId: WebAuthn.ToBase64Url(rawId),
             type: credential.type,
             clientExtensionResults: credential.getClientExtensionResults(),
             response: {
-                authenticatorData: WebAuthn.binaryToString(response.authenticatorData),
-                clientDataJSON: WebAuthn.binaryToString(response.clientDataJSON),
-                userHandle: response.userHandle && response.userHandle.byteLength > 0 ? WebAuthn.binaryToString(response.userHandle) : undefined,
-                signature: WebAuthn.binaryToString(response.signature)
+                authenticatorData: WebAuthn.ToBase64Url(authenticatorData),
+                clientDataJSON: WebAuthn.ToBase64Url(clientDataJSON),
+                userHandle: userHandle !== null ? WebAuthn.ToBase64Url(userHandle) : null,
+                signature: WebAuthn.ToBase64Url(signature)
             }
         }
-        console.log('result:', result)
+
+        console.log('result:', result);
+
         return result;
     }
 
 
 
-    private static binaryToString(value: ArrayBuffer): string {
-        return btoa(String.fromCharCode(...new Uint8Array(value))).replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
-    }
+    //private static stringToBinary(value: string): Uint8Array {
+    //    return Uint8Array.from(atob(value.replace(/-/g, "+").replace(/_/g, "/")), c => c.charCodeAt(0));
+    //}
 
-    private static stringToBinary(value: string): Uint8Array {
-        return Uint8Array.from(atob(value.replace(/-/g, "+").replace(/_/g, "/")), c => c.charCodeAt(0));
-    }
+    //private static binaryToString(value: ArrayBuffer): string {
+    //    return btoa(String.fromCharCode(...new Uint8Array(value))).replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
+    //}
 
-    private static base64ToString(value: string): string {
-        return value.replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
-    }
+    //private static base64ToString(value: string): string {
+    //    return value.replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
+    //}
+
+    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;
+    };
+
+    private static ToBase64Url(value: any) {
+        // Array or ArrayBuffer to Uint8Array
+        if (Array.isArray(value)) {
+            value = Uint8Array.from(value);
+        }
+
+        if (value instanceof ArrayBuffer) {
+            value = new Uint8Array(value);
+        }
+
+        // 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;
+    };
 }
 
 (window as any).WebAuthn = WebAuthn;
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 2fac870fe1..7fb0c7f0a6 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
@@ -31,18 +31,26 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
             DisplayName = user.DisplayName,
         };
 
+        var authenticatorSelection = new AuthenticatorSelection
+        {
+            ResidentKey = ResidentKeyRequirement.Required,
+            UserVerification = UserVerificationRequirement.Preferred
+        };
+
+        var extensions = new AuthenticationExtensionsClientInputs
+        {
+            CredProps = true,
+            Extensions = true,
+            UserVerificationMethod = true,
+        };
+
         var options = fido2.RequestNewCredential(new RequestNewCredentialParams
         {
             User = fidoUser,
             ExcludeCredentials = [.. existingKeys],
-            AuthenticatorSelection = AuthenticatorSelection.Default, // new() { AuthenticatorAttachment = AuthenticatorAttachment.Platform },
+            AuthenticatorSelection = authenticatorSelection,
             AttestationPreference = AttestationConveyancePreference.None,
-            Extensions = new AuthenticationExtensionsClientInputs
-            {
-                CredProps = true,
-                Extensions = true,
-                UserVerificationMethod = true,
-            }
+            Extensions = extensions
         });
 
         var key = GetWebAuthnCacheKey(userId);

From fac1b4cfeedea4d0330eaff900ecebede5bcd553 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 11:17:02 +0330
Subject: [PATCH 07/20] remove redundant stuff

---
 .../Layout/AppDiagnosticModal.razor.cs        |  1 -
 .../Scripts/WebAuthn.ts                       | 80 ++++++++-----------
 2 files changed, 34 insertions(+), 47 deletions(-)

diff --git a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
index d87d928e20..38944c9096 100644
--- a/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
+++ b/src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Core/Components/Layout/AppDiagnosticModal.razor.cs
@@ -1,7 +1,6 @@
 //#if (signalR == true)
 using Microsoft.AspNetCore.SignalR.Client;
 //#endif
-using Boilerplate.Shared.Controllers.Identity;
 using Boilerplate.Shared.Controllers.Diagnostics;
 using Boilerplate.Client.Core.Services.DiagnosticLog;
 
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 03b3db6307..59dd2a72dc 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,5 +1,5 @@
 class WebAuthn {
-    private static STORE_KEY = 'configured-webauthn';
+    private static STORE_KEY = 'webauthn';
 
     public static isAvailable() {
         return !!window.PublicKeyCredential;
@@ -11,14 +11,14 @@ class WebAuthn {
         localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(storedCredentials));
     }
 
-    public static isConfigured(username: string | undefined) {
+    public static removeConfigured(username: string) {
         const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
-        return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
+        localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(!!username ? storedCredentials.filter(c => c !== username) : []));
     }
 
-    public static removeConfigured(username: string) {
+    public static isConfigured(username: string | undefined) {
         const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
-        localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(!!username ? storedCredentials.filter(c => c !== username) : []));
+        return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
     }
 
 
@@ -138,17 +138,37 @@ class WebAuthn {
 
 
 
-    //private static stringToBinary(value: string): Uint8Array {
-    //    return Uint8Array.from(atob(value.replace(/-/g, "+").replace(/_/g, "/")), c => c.charCodeAt(0));
-    //}
+    private static ToBase64Url(value: any) {
+        // Array or ArrayBuffer to Uint8Array
+        if (Array.isArray(value)) {
+            value = Uint8Array.from(value);
+        }
 
-    //private static binaryToString(value: ArrayBuffer): string {
-    //    return btoa(String.fromCharCode(...new Uint8Array(value))).replace(/\+/g, "-").replace(/\//g, "_").replace(/=*$/g, "");
-    //}
+        if (value instanceof ArrayBuffer) {
+            value = new Uint8Array(value);
+        }
 
-    //private static base64ToString(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") {
@@ -181,38 +201,6 @@ class WebAuthn {
 
         return value;
     };
-
-    private static ToBase64Url(value: any) {
-        // Array or ArrayBuffer to Uint8Array
-        if (Array.isArray(value)) {
-            value = Uint8Array.from(value);
-        }
-
-        if (value instanceof ArrayBuffer) {
-            value = new Uint8Array(value);
-        }
-
-        // 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;
-    };
 }
 
 (window as any).WebAuthn = WebAuthn;

From 25d2f0f5f2bdc7ba0fdf918b0a7d2b5051efee85 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 11:45:07 +0330
Subject: [PATCH 08/20] remove redundant codes

---
 .../Scripts/WebAuthn.ts                       | 49 ++-----------------
 1 file changed, 5 insertions(+), 44 deletions(-)

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 59dd2a72dc..60144ae5b1 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
@@ -5,6 +5,11 @@ class WebAuthn {
         return !!window.PublicKeyCredential;
     }
 
+    public static isConfigured(username: string | undefined) {
+        const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
+        return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
+    }
+
     public static storeConfigured(username: string) {
         const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
         storedCredentials.push(username);
@@ -16,30 +21,13 @@ class WebAuthn {
         localStorage.setItem(WebAuthn.STORE_KEY, JSON.stringify(!!username ? storedCredentials.filter(c => c !== username) : []));
     }
 
-    public static isConfigured(username: string | undefined) {
-        const storedCredentials = JSON.parse(localStorage.getItem(WebAuthn.STORE_KEY) || '[]') as string[];
-        return !!username ? storedCredentials.includes(username) : storedCredentials.length > 0;
-    }
-
 
     public static async createCredential(options: PublicKeyCredentialCreationOptions) {
-        console.log(options);
-
-        //if (typeof options.challenge === 'string') {
-        //    options.challenge = WebAuthn.stringToBinary(options.challenge);
-        //}
-
         options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
 
-        //if (typeof options.user.id === 'string') {
-        //    options.user.id = WebAuthn.stringToBinary(options.user.id);
-        //}
-
         options.user.id = WebAuthn.ToArrayBuffer(options.user.id, 'user.id');
 
         for (let cred of options.excludeCredentials || []) {
-            //if (typeof cred.id !== 'string') continue;
-            //cred.id = WebAuthn.stringToBinary(cred.id);
             cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
         }
 
@@ -51,12 +39,8 @@ class WebAuthn {
             options.rp.id = undefined;
         }
 
-        console.log('corrected:', options);
-
         const credential = await navigator.credentials.create({ publicKey: options }) as PublicKeyCredential;
 
-        console.log('credential:', credential);
-
         const response = credential.response as AuthenticatorAttestationResponse;
 
         // Move data into Arrays incase it is super long
@@ -76,39 +60,18 @@ class WebAuthn {
             }
         };
 
-        console.log('result:', result);
-
         return result;
     }
 
     public static async verifyCredential(options: PublicKeyCredentialRequestOptions) {
-        console.log(options);
-
-        //if (typeof options.challenge === 'string') {
-        //    options.challenge = WebAuthn.stringToBinary(options.challenge);
-        //}
-
         options.challenge = WebAuthn.ToArrayBuffer(options.challenge, 'challenge');
 
-        //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.stringToBinary(id);
-        //        }
-        //    }
-        //}
-
         options.allowCredentials?.forEach(function (cred) {
             cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
         });
 
-        console.log('corrected:', options);
-
         const credential = await navigator.credentials.get({ publicKey: options }) as PublicKeyCredential;
 
-        console.log('credential:', credential);
-
         const response = credential.response as AuthenticatorAssertionResponse;
 
         // Move data into Arrays incase it is super long
@@ -131,8 +94,6 @@ class WebAuthn {
             }
         }
 
-        console.log('result:', result);
-
         return result;
     }
 

From 207c10c5e8d5b7edeef7d17e0bfd00bb00f87291 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 11:46:53 +0330
Subject: [PATCH 09/20] rename store key

---
 .../src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts      | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

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 60144ae5b1..ab1b60354b 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,5 +1,5 @@
 class WebAuthn {
-    private static STORE_KEY = 'webauthn';
+    private static STORE_KEY = 'bit-webauthn';
 
     public static isAvailable() {
         return !!window.PublicKeyCredential;

From 5a6d191fb85bf34916b037aa3a1c712626385c72 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 11:50:32 +0330
Subject: [PATCH 10/20] merge clear webauth button in diag modal

---
 .../Components/Layout/AppDiagnosticModal.razor | 10 ++--------
 .../Layout/AppDiagnosticModal.razor.Utils.cs   | 18 ++++++++----------
 2 files changed, 10 insertions(+), 18 deletions(-)

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 43cd6d8504..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,16 +88,10 @@
                                IconName="@BitIconName.RecycleBin" />
                 }
                 <BitButton IconOnly AutoLoading
-                           OnClick="ClearCache"
-                           Title="Clear cache"
+                           OnClick="ClearData"
+                           Title="Clear data"
                            Color="BitColor.SecondaryBackground"
                            IconName="@BitIconName.Clear" />
-                <BitButton IconOnly AutoLoading
-                           Color="BitColor.Info"
-                           Variant="BitVariant.Text"
-                           IconName="@BitIconName.Delete"
-                           Title="Clear WebAuthn"
-                           OnClick="ClearWebAuthn" />
             </BitStack>
 
             <BitBasicList @ref="logStackRef"
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 384590bfe3..cc7d9e607e 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
@@ -105,7 +105,7 @@ private string GetMemoryUsage()
         return $"{memory / (1024.0 * 1024.0):F2} MB";
     }
 
-    private async Task ClearCache()
+    private async Task ClearData()
     {
         try
         {
@@ -120,6 +120,13 @@ private async Task ClearCache()
             await cookie.Remove(item.Name!);
         }
 
+        if ((await AuthenticationStateTask).User.IsAuthenticated())
+        {
+            await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
+
+            await JSRuntime.RemoveWebAuthnConfigured();
+        }
+
         if (AppPlatform.IsBlazorHybrid is false)
         {
             await JSRuntime.InvokeVoidAsync("BitBswup.forceRefresh"); // Clears cache storages and uninstalls service-worker.
@@ -129,13 +136,4 @@ private async Task ClearCache()
             NavigationManager.Refresh(forceReload: true);
         }
     }
-
-    private async Task ClearWebAuthn()
-    {
-        if ((await AuthenticationStateTask).User.IsAuthenticated() is false) return;
-
-        await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
-
-        await JSRuntime.RemoveWebAuthnConfigured();
-    }
 }

From 3f79a37d084aa83de29ed9765b197f089550cf75 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 12:39:09 +0330
Subject: [PATCH 11/20] remove existing keys from create option

---
 .../Controllers/Identity/UserController.WebAuthn.cs             | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

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 7fb0c7f0a6..d1afe01211 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
@@ -47,7 +47,7 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
         var options = fido2.RequestNewCredential(new RequestNewCredentialParams
         {
             User = fidoUser,
-            ExcludeCredentials = [.. existingKeys],
+            ExcludeCredentials = [], //[.. existingKeys],
             AuthenticatorSelection = authenticatorSelection,
             AttestationPreference = AttestationConveyancePreference.None,
             Extensions = extensions

From 8b7330b17e69153bba50f999e7edc983d21b5c0f Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Wed, 12 Mar 2025 16:54:32 +0330
Subject: [PATCH 12/20] fix tfa issues

---
 .../Pages/Identity/SignIn/SignInPage.razor.cs | 44 +++++++++++++------
 .../Identity/IdentityController.WebAuthn.cs   | 31 +++++++++----
 .../Identity/IdentityController.cs            | 11 ++++-
 .../Identity/UserController.WebAuthn.cs       |  2 +-
 .../Identity/IIdentityController.cs           |  5 ++-
 .../src/Shared/Dtos/AppJsonContext.cs         |  2 +
 .../Identity/VerifyWebAuthnAndSignInDto.cs    | 10 +++++
 7 files changed, 80 insertions(+), 25 deletions(-)
 create mode 100644 src/Templates/Boilerplate/Bit.Boilerplate/src/Shared/Dtos/Identity/VerifyWebAuthnAndSignInDto.cs

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..d85acfe039 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!;
 
 
@@ -105,20 +106,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 +170,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 +181,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);
             }
@@ -242,9 +253,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/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 f677138a13..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
@@ -34,7 +34,7 @@ public async Task<AssertionOptions> 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<VerifyAssertionResult> VerifyWebAuthAssertion(AuthenticatorAss
     }
 
     [HttpPost, Produces<SignInResponseDto>()]
-    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 d1afe01211..05709f5efb 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
@@ -54,7 +54,7 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
         });
 
         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;
     }
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<VerifyAssertionResult> VerifyWebAuthAssertion(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken);
 
     [HttpPost]
-    Task<SignInResponseDto> VerifyWebAuthAndSignIn(AuthenticatorAssertionRawResponse clientResponse, CancellationToken cancellationToken) => default!;
+    Task<SignInResponseDto> VerifyWebAuthAndSignIn(VerifyWebAuthnAndSignInDto request, CancellationToken cancellationToken) => default!;
+
+    [HttpPost]
+    Task VerifyWebAuthAndSendTwoFactorToken(AuthenticatorAssertionRawResponse clientResponse, 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; }
+}

From 542550395e5d5c5ea76dfb64bf0a3e6b3281966e Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 07:17:45 +0330
Subject: [PATCH 13/20] reduce authenticator selection options

---
 .../Pages/Identity/SignIn/SignInPage.razor.cs         |  2 ++
 .../Controllers/Identity/UserController.WebAuthn.cs   | 11 ++++++++---
 2 files changed, 10 insertions(+), 3 deletions(-)

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 d85acfe039..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
@@ -84,6 +84,7 @@ protected override async Task OnInitAsync()
 
             if (source == TfaPayload)
             {
+                webAuthnAssertion = null;
                 requiresTwoFactor = false;
                 model.TwoFactorCode = null;
             }
@@ -197,6 +198,7 @@ private async Task HandleOnPasswordlessSignIn()
         }
         catch (KnownException e)
         {
+            webAuthnAssertion = null;
             SnackBarService.Error(e.Message);
         }
         finally
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 05709f5efb..49930b4171 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
@@ -31,10 +31,15 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
             DisplayName = user.DisplayName,
         };
 
+        //var authenticatorSelection = new AuthenticatorSelection
+        //{
+        //    ResidentKey = ResidentKeyRequirement.Required,
+        //    UserVerification = UserVerificationRequirement.Preferred
+        //};
+
         var authenticatorSelection = new AuthenticatorSelection
         {
-            ResidentKey = ResidentKeyRequirement.Required,
-            UserVerification = UserVerificationRequirement.Preferred
+            AuthenticatorAttachment = AuthenticatorAttachment.Platform
         };
 
         var extensions = new AuthenticationExtensionsClientInputs
@@ -50,7 +55,7 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
             ExcludeCredentials = [], //[.. existingKeys],
             AuthenticatorSelection = authenticatorSelection,
             AttestationPreference = AttestationConveyancePreference.None,
-            Extensions = extensions
+            //Extensions = extensions
         });
 
         var key = GetWebAuthnCacheKey(userId);

From d5c9b4c560463ef46a5a0ba3c0d8563b31415ccb Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 07:59:36 +0330
Subject: [PATCH 14/20] reduce assertion options

---
 .../src/Client/Boilerplate.Client.Core/Scripts/WebAuthn.ts    | 4 ++--
 .../Controllers/Identity/IdentityController.WebAuthn.cs       | 4 ++--
 2 files changed, 4 insertions(+), 4 deletions(-)

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 ab1b60354b..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
@@ -27,9 +27,9 @@ class WebAuthn {
 
         options.user.id = WebAuthn.ToArrayBuffer(options.user.id, 'user.id');
 
-        for (let cred of options.excludeCredentials || []) {
+        options.excludeCredentials?.forEach(function (cred) {
             cred.id = WebAuthn.ToArrayBuffer(cred.id, 'cred.id');
-        }
+        });
 
         if (options.authenticatorSelection?.authenticatorAttachment === null) {
             options.authenticatorSelection.authenticatorAttachment = undefined;
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 6f7aadb423..21485f8886 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
@@ -28,8 +28,8 @@ public async Task<AssertionOptions> GetWebAuthnAssertionOptions(CancellationToke
 
         var options = fido2.GetAssertionOptions(new GetAssertionOptionsParams
         {
-            Extensions = extensions,
-            AllowedCredentials = existingKeys,
+            //Extensions = extensions,
+            //AllowedCredentials = existingKeys,
             UserVerification = UserVerificationRequirement.Discouraged,
         });
 

From 53330a291103eaac3c80eb763cd8eb922811a833 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 08:50:44 +0330
Subject: [PATCH 15/20] restore extensions options back

---
 .../Controllers/Identity/IdentityController.WebAuthn.cs       | 4 ++--
 .../Controllers/Identity/UserController.WebAuthn.cs           | 2 +-
 2 files changed, 3 insertions(+), 3 deletions(-)

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 21485f8886..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
@@ -28,8 +28,8 @@ public async Task<AssertionOptions> GetWebAuthnAssertionOptions(CancellationToke
 
         var options = fido2.GetAssertionOptions(new GetAssertionOptionsParams
         {
-            //Extensions = extensions,
-            //AllowedCredentials = existingKeys,
+            Extensions = extensions,
+            AllowedCredentials = existingKeys,
             UserVerification = UserVerificationRequirement.Discouraged,
         });
 
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 49930b4171..df60dfe2fc 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
@@ -55,7 +55,7 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
             ExcludeCredentials = [], //[.. existingKeys],
             AuthenticatorSelection = authenticatorSelection,
             AttestationPreference = AttestationConveyancePreference.None,
-            //Extensions = extensions
+            Extensions = extensions
         });
 
         var key = GetWebAuthnCacheKey(userId);

From 807332be091a4c48d7311312056cc3cc6dd24c09 Mon Sep 17 00:00:00 2001
From: ysmoradi <ysmoradi@outlook.com>
Date: Thu, 13 Mar 2025 07:31:08 +0100
Subject: [PATCH 16/20] trusted origins

---
 .../Program.Services.cs                       | 34 +++++++++++--------
 .../ServerApiSettings.cs                      | 10 +++---
 .../Boilerplate.Server.Api/appsettings.json   |  4 +--
 3 files changed, 26 insertions(+), 22 deletions(-)

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..a9484f56ea 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<string>(trustedOrigins.Select(uri => uri.AbsoluteUri));
-            options.ServerIcon = "";
+
+        });
+
+        services.AddScoped(sp =>
+        {
+            var webAppUrl = sp.GetRequiredService<IHttpContextAccessor>()
+                .HttpContext!.Request.GetWebAppUrl();
+
+            var options = new Fido2Configuration
+            {
+                TimestampDriftTolerance = 1000,
+                ServerName = "Boilerplate WebAuthn",
+                ServerDomain = webAppUrl.Host,
+                Origins = new HashSet<string>([webAppUrl.AbsoluteUri]),
+                ServerIcon = ""
+            };
+
+            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!;
 
     /// <summary>
-    /// 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.
     /// </summary>
-    public Uri[] AllowedOrigins { get; set; } = [];
+    public Uri[] TrustedOrigins { get; set; } = [];
 
     //#if (module == "Admin" || module == "Sales")
     [Required]
@@ -113,8 +113,8 @@ public override IEnumerable<ValidationResult> 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`).",

From 43e3625bbd12a87454b2ad1f41e2ee94ce1c3529 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 11:20:10 +0330
Subject: [PATCH 17/20] change servericon value to url

---
 .../wwwroot/images/icons/bit-logo.png            | Bin 0 -> 2463 bytes
 .../Identity/UserController.WebAuthn.cs          |  14 +++++++-------
 .../Boilerplate.Server.Api/Program.Services.cs   |   5 +++--
 3 files changed, 10 insertions(+), 9 deletions(-)
 create mode 100644 src/Templates/Boilerplate/Bit.Boilerplate/src/Client/Boilerplate.Client.Web/wwwroot/images/icons/bit-logo.png

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 0000000000000000000000000000000000000000..1629cbe86b5305df2fb933cbae2cfb6edde90fab
GIT binary patch
literal 2463
zcmV;Q31Ie#P)<h;3K|Lk000e1NJLTq003S9002J-1^@s6X?lN=00009a7bBm001r{
z001r{0eGc9b^rhX8FWQhbW?9;ba!ELWdL_~cP?peYja~^aAhuUa%Y?FJQ@H12`ouO
zK~#90?VNvbRMi#7Kj-ZRA|%<j8DNURR7Z_Ng*sG6#-_E_+Tt))`(tFLwFN8L1VpGm
zI;|aAEcA!1RjdwRb~pSGr%EZ)c1nL}k+$Lh3QldOl?v1Xr9!DB`xb~ol6|LtBnZLm
z?tQ!O?V6I$OlGoq=broBllShPd+xm}s)`6gQ~EYUXM<h}(uK+vLVCZp+M5`{cLY?3
zIV*r$2e_U=rA3pG<t5hje~-}Q%tW*;hfOz`&_rEcy|ZMh&H}97;Uy;(gnF7vqUJ2Z
z>K)Vbpw@&EXgRB}Kq~WEiJnseEoT*0xdoi93<bVS3<bVS3<bVS3<bVS3<bVSg#C6X
zbKa<;XZWRP<^r(BAzFoMB_hXB+CdIM<`txNm&WP#yMH5%Evb53u0~`SB7fF+<WB|r
zYChV4W*U26gf?rUdY4f<VktGSdrlMfH3K>jgr<(kh<qC48sGz%{X<UxFCh9fqQBJE
z+8ri>i$N9==5^;m;0_>k+NBNjgm2LmRd0Cnm3{<t7N<rI7o-)5+?nsDDmd^m``8#V
zJ+puqFwRtPR#dhknn3Iy2lXUKv$GhuD~o^^aiWd7mW-LN^!Gr&KCmnKz9u6BUpv{}
zYv85j=qTLGHxXKZ;U*$IvIkg1oxNJ`!}ZIl%shN9q3}L(J&h_iYcjGfudgrdt)wE7
z%0BDY<8-H^y7K(v*?bs(xQZas?94?o|3H{WFhK{n09eWE&Q@u1Ci&(yT+v9rAyD|J
zk*{HqF}X`EQc0!#o?I?W13#LWvk`a{h#0mhmMgK?CM~IZMUdj%xfTA{0G8yBTUTxy
z)4;zE4DSYi38Qj9G3Vzp{m{swJZbo)yf$KOE2<9<43<3pCMt(NBQ2?dpSHd<@25FT
z_Q0D@7V6(CgC+0)jVRAZ%fS%^d#H4qKz}#DmDjM`8#F%l&BEn*V=x6CkjZe)6SA-{
z-TG+l5luv=pz;aO2Y^R_>2y02N->W(9p7iJz(HUG=!;m^PBJ*Cdjdiu5Ut132`JYB
zAEf^z#th{%9B}4?{p~^pYBI8+SK~?SLR}S%X&!HYK7(2hF)p&vB>(QqtDAs(fEtgT
zb%WKYiJI*NNi7nY?VL}BJvogsa-lAd{4?8SKZcrJ2-j*Ndc7ti8%!rXx_s0DP1uXj
ziUvdzfR`Jf0&(Lqt$-)yT|#kzS9ub7Ym&z8KNRdDIr5Okqb*qORKU1af-(s^eOsaZ
zmYc$d8qptWJo;5%#msK6!wo%+V!EqgA7WqZrM4q84_6ZoeF4{+h^3XFyU55>+Gg(?
z(0&@jh^BAC)yY^|4S-S&^10kS=^BJ1eO|hO*XVZF>5AI@P;lU7nI<B0ytgy(+~m9$
zi*3LdqXx<hP1>u?e5XE%B`bhYXI$z+_>RW=e_%^2^?pQt23%;QB|qK3y_&EW7ZUjA
zsI#xr`{?mW%sY@-9arI6n=s3|HfzHEP+p%chsUDf-9Yvku&Xy}TXa2u+@6lo%??C!
z)?5lg^cJsUd?z7p4KnBe+FH96WtpLQqs%&Taek+vYsnRK0FfKt{yn2#3?%3PDqr&!
zc;P{P%-$P_m@}!6ZvlsmY%}_){6?0+j8{(CC>I&^T@c#fCGeee@uR@BoPjQjq!2ah
zPxN}<1xTJey*THunFn52t1E3MI57kA1Q^W=Cd)#fXW<p0)tEkU-g(eG@Gid!NJLgN
z9{m^aoPp^`4t6y7+V_&`cCe-*?Dx@a24R6F>{X_BiDKzS9|d?}GbU>RZumUF*Fu*j
zYPQSW|GtFIs%f|~8PVP}RCOdGQ*$?#-yquA_v$TRH<tAiZLQf2=0&mt8jq$6)L$H`
zI}wuwD6J2q`^Dql0v_-C0Ss$3E^~FcU39Hxuc4j)Xvk=v3aaM?ti)GL1Aih=6}bqz
zW_?>!2du<bOat%3VsnA0m{lN^V*xAi6$`Vwxh`lSS#e#nMeb7{WP<_=(Y^W2wBGkT
z*%ZsLQt4XeCFOSk7AgiQm{jt5$Cbg$835%nGduZfz)F0@Fz|p{{!h#EAPceb=F7cg
z2zCA~a6vv|A+DKVVMfW9+76_zLorDcca(B#h96k;RBQyJ9kTUAoST9c)DP9UhK9cl
zR*3&teS5;U7+DKHka>Y^R_)(aEd6od1~X&U2dl<^PUi8yFtopj6REERCFUeD{m@86
zml&EK<lN{BfeH$M-oU%oD&VM53zda3EB&#c1c@M24qE}3RvTM`UJ(ld3qadyIuLC&
zY!kxOrzF-sJ`hE*)B_;X46VDU=uQNpBVbOI48lE&0n^N2G!oh<vGnDB36XgWmzc8(
z#KTagD|PAU|M;b$l)N4I-ZvnNJhUB)=q8D!<`jEyKw|CVIqv)(VTOlh9Z2{a#q?WF
z`WMM0vKa6(0~~?K5@ODFX-=7CAg6aX``8$1aUMoOf1!61s|R$x#;b!`nhTtM^%srP
zElpMvmis&~%!8lHLDoyNvkO>*=sHr-9Rr=9>pNrGh{)%GDS+AWm;U^ky6B)B+fVY2
z!k4zyY?D~}_B>ygm5l~q0V)fyofFc`tDM>oIUE=dGOEZT^B!Dlx}PTk%fWyZM@>eS
zOS3Z(_;L}xhl62(%bYYWpVhXS4xjy&oa_rBN&7B@r6tsok2GRU(Kh?#atrZ;tbtcW
zN!W8iOt;bnhtiI^*K54y#d1sWi=0b;swm0G9SC<*u60-CCBj^x$=X3z7RW{3HCI(s
z<MzY2Tm|xKDMa<*A}poH`EbehYnBk>on+c(zl;;P47d{rdXLR^)NRzbeMheE1Mq}s
zmoGK|ZYcl4x`uMrzC7&~*0pqL!hVEMxPjiIihlnCwxDt|Nqe&ILyLyCl7HRdc_UZx
z>cgmP&2Awt<ogPi!p@66R`p*^*mqKOY%IzQ;Ezz?SDgs^5lx_Sg(mDqO-9zKdOC<^
zMBmBod&nAmSwo$@%GjY;w=&SCR;3-RyrmCO_2)gx^g|;#r?L_F2&xT0Ju3Bxp1yF>
z3G4>l1@d=rpVdU|E4~OY)j9L%C4@6`ahV8=z~WU@9@j+VS<k*&j*f!Ne3Ywu|G*{)
deb=Ov@jvdHCuR^631t8P002ovPDHLkV1gB@mX-hj

literal 0
HcmV?d00001

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 df60dfe2fc..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
@@ -31,17 +31,17 @@ public async Task<CredentialCreateOptions> GetWebAuthnCredentialOptions(Cancella
             DisplayName = user.DisplayName,
         };
 
-        //var authenticatorSelection = new AuthenticatorSelection
-        //{
-        //    ResidentKey = ResidentKeyRequirement.Required,
-        //    UserVerification = UserVerificationRequirement.Preferred
-        //};
-
         var authenticatorSelection = new AuthenticatorSelection
         {
-            AuthenticatorAttachment = AuthenticatorAttachment.Platform
+            ResidentKey = ResidentKeyRequirement.Required,
+            UserVerification = UserVerificationRequirement.Preferred
         };
 
+        //var authenticatorSelection = new AuthenticatorSelection
+        //{
+        //    AuthenticatorAttachment = AuthenticatorAttachment.Platform
+        //};
+
         var extensions = new AuthenticationExtensionsClientInputs
         {
             CredProps = true,
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 a9484f56ea..d713868757 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
@@ -222,7 +222,8 @@ void AddDbContext(DbContextOptionsBuilder options)
             //#elif (database == "Other")
             throw new NotImplementedException("Install and configure any database supported by ef core (https://learn.microsoft.com/en-us/ef/core/providers)");
             //#endif
-        };
+        }
+        ;
 
         services.AddOptions<IdentityOptions>()
             .Bind(configuration.GetRequiredSection(nameof(ServerApiSettings.Identity)))
@@ -315,7 +316,7 @@ void AddDbContext(DbContextOptionsBuilder options)
                 ServerName = "Boilerplate WebAuthn",
                 ServerDomain = webAppUrl.Host,
                 Origins = new HashSet<string>([webAppUrl.AbsoluteUri]),
-                ServerIcon = ""
+                ServerIcon = new Uri(webAppUrl, "images/icons/bit-logo.png").ToString()
             };
 
             return options;

From 897c9473b80814116cdd323e5584a10540e25d24 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 11:34:26 +0330
Subject: [PATCH 18/20] remove redundant storage item removal

---
 .../Components/Layout/AppDiagnosticModal.razor.Utils.cs         | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

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 cc7d9e607e..ba9a535897 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
@@ -124,7 +124,7 @@ private async Task ClearData()
         {
             await userController.DeleteAllWebAuthnCredentials(CurrentCancellationToken);
 
-            await JSRuntime.RemoveWebAuthnConfigured();
+            //await JSRuntime.RemoveWebAuthnConfigured();
         }
 
         if (AppPlatform.IsBlazorHybrid is false)

From 8eca1c80bfa1b7e0fa2fc74aab89d1d1fba4bb7a Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 13:13:49 +0330
Subject: [PATCH 19/20] fix review comments

---
 .../Components/Layout/AppDiagnosticModal.razor.Utils.cs        | 2 ++
 .../src/Server/Boilerplate.Server.Api/Program.Services.cs      | 3 +--
 2 files changed, 3 insertions(+), 2 deletions(-)

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 ba9a535897..a727ff7353 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
@@ -124,6 +124,8 @@ private async Task ClearData()
         {
             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();
         }
 
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 d713868757..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
@@ -222,8 +222,7 @@ void AddDbContext(DbContextOptionsBuilder options)
             //#elif (database == "Other")
             throw new NotImplementedException("Install and configure any database supported by ef core (https://learn.microsoft.com/en-us/ef/core/providers)");
             //#endif
-        }
-        ;
+        };
 
         services.AddOptions<IdentityOptions>()
             .Bind(configuration.GetRequiredSection(nameof(ServerApiSettings.Identity)))

From a3d6765627b53fa6a6d22920f6779180099ab167 Mon Sep 17 00:00:00 2001
From: Saleh Yusefnejad <msynk@outlook.com>
Date: Thu, 13 Mar 2025 13:14:52 +0330
Subject: [PATCH 20/20] fix typo

---
 .../Components/Layout/AppDiagnosticModal.razor.Utils.cs         | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

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 a727ff7353..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
@@ -124,7 +124,7 @@ private async Task ClearData()
         {
             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.
+            // 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();
         }