feat(security): AutoLoginAuthenticationHandler for dev login bypass

This commit is contained in:
Joseph Doherty
2026-06-11 04:31:07 -04:00
parent a27e82c8d1
commit caeaae21f9
2 changed files with 120 additions and 0 deletions
@@ -0,0 +1,64 @@
using System.Text.Encodings.Web;
using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Authentication.Cookies;
using Microsoft.AspNetCore.Http;
using Microsoft.Extensions.Logging.Abstractions;
using Microsoft.Extensions.Options;
using Shouldly;
using Xunit;
using ZB.MOM.WW.OtOpcUa.Security.Auth;
namespace ZB.MOM.WW.OtOpcUa.Security.Tests;
public class AutoLoginAuthenticationHandlerTests
{
private static async Task<AuthenticateResult> AuthenticateAsync(string user = "multi-role-test")
{
var schemeOpts = new StubOptionsMonitor<AuthenticationSchemeOptions>(new AuthenticationSchemeOptions());
var disableOpts = Options.Create(new AuthDisableLoginOptions { DisableLogin = true, User = user });
var handler = new AutoLoginAuthenticationHandler(
schemeOpts, NullLoggerFactory.Instance, UrlEncoder.Default, disableOpts);
await handler.InitializeAsync(
new AuthenticationScheme(
CookieAuthenticationDefaults.AuthenticationScheme, null, typeof(AutoLoginAuthenticationHandler)),
new DefaultHttpContext());
return await handler.AuthenticateAsync();
}
[Fact]
public async Task Authenticates_as_configured_user_with_all_roles()
{
var result = await AuthenticateAsync();
result.Succeeded.ShouldBeTrue();
result.Principal!.Identity!.IsAuthenticated.ShouldBeTrue();
result.Principal.Identity.Name.ShouldBe("multi-role-test");
foreach (var role in DevAuthRoles.All)
result.Principal.IsInRole(role).ShouldBeTrue();
// Satisfies the FleetAdmin (Administrator) + DriverOperator (Operator|Administrator) policies.
result.Principal.IsInRole("Administrator").ShouldBeTrue();
result.Principal.IsInRole("Operator").ShouldBeTrue();
}
[Fact]
public async Task Honours_configured_username()
{
var result = await AuthenticateAsync("custom-dev");
result.Principal!.Identity!.Name.ShouldBe("custom-dev");
}
/// <summary>
/// Minimal <see cref="IOptionsMonitor{TOptions}"/> stub returning a fixed value for any
/// name. The test project does not reference Moq, so the scheme-options monitor the base
/// <see cref="AuthenticationHandler{TOptions}"/> needs is hand-rolled here.
/// </summary>
private sealed class StubOptionsMonitor<T>(T value) : IOptionsMonitor<T>
{
public T CurrentValue { get; } = value;
public T Get(string? name) => CurrentValue;
public IDisposable? OnChange(Action<T, string?> listener) => null;
}
}