File: UserJwtsTests.cs
Web Access
Project: src\src\Tools\dotnet-user-jwts\test\dotnet-user-jwts.Tests.csproj (Microsoft.AspNetCore.Authentication.JwtBearer.Tools.Tests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims;
using System.Text.Json;
using System.Text.Json.Nodes;
using System.Text.RegularExpressions;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.InternalTesting;
using Microsoft.AspNetCore.TestHost;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Configuration.UserSecrets;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Tools.Internal;
using Xunit.Abstractions;
 
namespace Microsoft.AspNetCore.Authentication.JwtBearer.Tools.Tests;
 
public class UserJwtsTests(UserJwtsTestFixture fixture, ITestOutputHelper output) : IClassFixture<UserJwtsTestFixture>
{
    private readonly TestConsole _console = new(output);
 
    [Fact]
    public void List_NoTokensForNewProject()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "list", "--project", project });
        Assert.Contains("No JWTs created yet!", _console.GetOutput());
    }
 
    [Fact]
    public void List_HandlesNoSecretsInProject()
    {
        var project = Path.Combine(fixture.CreateProject(false), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "list", "--project", project });
        Assert.DoesNotContain("Set UserSecretsId to ", _console.GetOutput());
        Assert.Contains("No JWTs created yet!", _console.GetOutput());
    }
 
    [Fact]
    public void Create_CreatesSecretOnNoSecretInproject()
    {
        var project = Path.Combine(fixture.CreateProject(false), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        var output = _console.GetOutput();
        Assert.DoesNotContain("could not find SecretManager.targets", output);
        Assert.DoesNotContain("Set UserSecretsId to ", output);
        Assert.Contains("New JWT saved", output);
    }
 
    [Fact]
    public void Create_WritesGeneratedTokenToDisk()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Development.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        Assert.Contains("New JWT saved", _console.GetOutput());
        Assert.Contains("dotnet-user-jwts", File.ReadAllText(appsettings));
    }
 
    [Fact]
    public async Task Create_TokenAcceptedByJwtBearerHandler()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Development.json");
        var secrets = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
        var app = new Program(_console);
 
        app.Run(["create", "--project", project, "-o", "token"]);
        var token = _console.GetOutput().Trim();
 
        var builder = WebApplication.CreateEmptyBuilder(new());
        builder.WebHost.UseTestServer();
 
        builder.Configuration.AddJsonFile(appsettings);
        builder.Configuration.AddJsonFile(secrets);
 
        builder.Services.AddRouting();
        builder.Services.AddAuthentication().AddJwtBearer();
        builder.Services.AddAuthorization();
 
        using var webApp = builder.Build();
        webApp.MapGet("/secret", (ClaimsPrincipal user) => $"Hello {user.Identity?.Name}!")
            .RequireAuthorization();
 
        await webApp.StartAsync();
 
        var client = webApp.GetTestClient();
        client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token}");
        Assert.Equal($"Hello {Environment.UserName}!", await client.GetStringAsync("/secret"));
    }
 
    [Fact]
    public void Create_CanModifyExistingScheme()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Development.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        var appSettings = JsonSerializer.Deserialize<JsonObject>(File.ReadAllText(appsettings));
        Assert.Equal("dotnet-user-jwts", appSettings["Authentication"]["Schemes"]["Bearer"]["ValidIssuer"].GetValue<string>());
        app.Run(["create", "--project", project, "--issuer", "new-issuer"]);
        appSettings = JsonSerializer.Deserialize<JsonObject>(File.ReadAllText(appsettings));
        Assert.Equal("new-issuer", appSettings["Authentication"]["Schemes"]["Bearer"]["ValidIssuer"].GetValue<string>());
    }
 
    [Fact]
    public void Create_CanModifyExistingSchemeInGivenAppSettings()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Local.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--appsettings-file", "appsettings.Local.json" });
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        var appSettings = JsonSerializer.Deserialize<JsonObject>(File.ReadAllText(appsettings));
        Assert.Equal("dotnet-user-jwts", appSettings["Authentication"]["Schemes"]["Bearer"]["ValidIssuer"].GetValue<string>());
        app.Run(["create", "--project", project, "--issuer", "new-issuer", "--appsettings-file", "appsettings.Local.json"]);
        appSettings = JsonSerializer.Deserialize<JsonObject>(File.ReadAllText(appsettings));
        Assert.Equal("new-issuer", appSettings["Authentication"]["Schemes"]["Bearer"]["ValidIssuer"].GetValue<string>());
    }
 
    [Fact]
    public void Print_ReturnsNothingForMissingToken()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "print", "invalid-id", "--project", project });
        Assert.Contains("No token with ID 'invalid-id' found", _console.GetOutput());
    }
 
    [Fact]
    public void List_ReturnsIdForGeneratedToken()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--scheme", "MyCustomScheme" });
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        app.Run(new[] { "list", "--project", project });
        Assert.Contains("MyCustomScheme", _console.GetOutput());
    }
 
    [Fact]
    public void List_ReturnsIdForGeneratedToken_WithJsonFormat()
    {
        var schemeName = "MyCustomScheme";
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--scheme", schemeName });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
        _console.ClearOutput();
 
        app.Run(new[] { "list", "--project", project, "--output", "json" });
        var output = _console.GetOutput();
        var deserialized = JsonSerializer.Deserialize<Dictionary<string, Jwt>>(output);
 
        var jwt = deserialized[id];
 
        Assert.NotNull(deserialized);
        Assert.Equal(schemeName, jwt.Scheme);
    }
 
    [Fact]
    public void List_ReturnsEmptyListWhenNoTokens_WithJsonFormat()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "list", "--project", project, "--output", "json" });
        var output = _console.GetOutput();
 
        Assert.Equal("[]", output.Trim());
    }
 
    [Fact]
    public void Remove_RemovesGeneratedToken()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Development.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
        app.Run(new[] { "create", "--project", project, "--scheme", "Scheme2" });
 
        app.Run(new[] { "remove", id, "--project", project });
        var appsettingsContent = File.ReadAllText(appsettings);
        Assert.DoesNotContain(DevJwtsDefaults.Scheme, appsettingsContent);
        Assert.Contains("Scheme2", appsettingsContent);
    }
 
    [Fact]
    public void Remove_RemovesGeneratedTokenInGivenAppsettings()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Local.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--appsettings-file", "appsettings.Local.json" });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
        app.Run(new[] { "create", "--project", project, "--appsettings-file", "appsettings.Local.json", "--scheme", "Scheme2" });
 
        app.Run(new[] { "remove", id, "--project", project, "--appsettings-file", "appsettings.Local.json" });
        var appsettingsContent = File.ReadAllText(appsettings);
        Assert.DoesNotContain(DevJwtsDefaults.Scheme, appsettingsContent);
        Assert.Contains("Scheme2", appsettingsContent);
    }
 
    [Fact]
    public void Clear_RemovesGeneratedTokens()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Development.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        app.Run(new[] { "create", "--project", project, "--scheme", "Scheme2" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        app.Run(new[] { "clear", "--project", project, "--force" });
        var appsettingsContent = File.ReadAllText(appsettings);
        Assert.DoesNotContain(DevJwtsDefaults.Scheme, appsettingsContent);
        Assert.DoesNotContain("Scheme2", appsettingsContent);
    }
 
    [Fact]
    public void Clear_RemovesGeneratedTokensInGivenAppsettings()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var appsettings = Path.Combine(Path.GetDirectoryName(project), "appsettings.Local.json");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--appsettings-file", "appsettings.Local.json" });
        app.Run(new[] { "create", "--project", project, "--appsettings-file", "appsettings.Local.json", "--scheme", "Scheme2" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        app.Run(new[] { "clear", "--project", project, "--appsettings-file", "appsettings.Local.json", "--force" });
        var appsettingsContent = File.ReadAllText(appsettings);
        Assert.DoesNotContain(DevJwtsDefaults.Scheme, appsettingsContent);
        Assert.DoesNotContain("Scheme2", appsettingsContent);
    }
 
    [Fact]
    public void Key_CanResetSigningKey()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        app.Run(new[] { "key", "--project", project });
        Assert.Contains("Signing Key:", _console.GetOutput());
 
        app.Run(new[] { "key", "--reset", "--force", "--project", project });
        Assert.Contains("New signing key created:", _console.GetOutput());
    }
 
    [Fact]
    public async Task Key_CanResetSigningKey_WhenSecretsHasPrepulatedData()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
        await File.WriteAllTextAsync(secretsFilePath,
@"{
  ""Foo"": {
    ""Bar"": ""baz""
  }
}"{
  ""Foo"": {
    ""Bar"": ""baz""
  }
}");
 
        app.Run(new[] { "create", "--project", project });
        app.Run(new[] { "key", "--project", project });
        Assert.Contains("Signing Key:", _console.GetOutput());
 
        app.Run(new[] { "key", "--reset", "--force", "--project", project });
        Assert.Contains("New signing key created:", _console.GetOutput());
 
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        Assert.NotNull(secretsJson);
        Assert.True(secretsJson.ContainsKey(SigningKeysHandler.GetSigningKeyPropertyName(DevJwtsDefaults.Scheme)));
        Assert.True(secretsJson.TryGetPropertyValue("Foo", out var fooField));
        Assert.Equal("baz", fooField["Bar"].GetValue<string>());
    }
 
    [Fact]
    public void Command_ShowsHelpForInvalidCommand()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        var exception = Record.Exception(() => app.Run(new[] { "not-real", "--project", project }));
 
        Assert.Null(exception);
        Assert.Contains("Unrecognized command or argument 'not-real'", _console.GetOutput());
    }
 
    [Fact]
    public void CreateCommand_ShowsBasicTokenDetails()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        var output = _console.GetOutput();
 
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains("Token: ", output);
        Assert.DoesNotContain("Scheme", output);
    }
 
    [Fact]
    public void CreateCommand_SupportsODateTimeFormats()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--expires-on", DateTime.Now.AddDays(2).ToString("O") });
        var output = _console.GetOutput();
 
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains("Token: ", output);
        Assert.Contains("Expires On", output);
        Assert.DoesNotContain("Scheme", output);
    }
 
    [Fact]
    public void CreateCommand_ShowsCustomizedTokenDetails()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--scheme", "customScheme" });
        var output = _console.GetOutput();
 
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains("Token: ", output);
        Assert.Contains("Scheme: customScheme", output);
    }
 
    [Fact]
    public void CreateCommand_DisplaysErrorForInvalidExpiresOnCombination()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--expires-on", DateTime.UtcNow.AddDays(2).ToString("O"), "--valid-for", "2h" });
        var output = _console.GetOutput();
 
        Assert.Contains($"'--valid-for' and '--expires-on' are mutually exclusive flags. Provide either option but not both.", output);
        Assert.DoesNotContain("Expires On: ", output);
    }
 
    [Fact]
    public void PrintCommand_ShowsBasicOptions()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
 
        app.Run(new[] { "print", id, "--project", project });
        var output = _console.GetOutput();
 
        Assert.Contains($"ID: {id}", output);
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains($"Scheme: {DevJwtsDefaults.Scheme}", output);
        Assert.Contains($"Audience(s): http://localhost:23528, https://localhost:44395, https://localhost:5001, http://localhost:5000", output);
    }
 
    [Fact]
    public void PrintCommand_ShowsBasicOptions_WithJsonFormat()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
        _console.ClearOutput();
 
        app.Run(new[] { "print", id, "--project", project, "--output", "json" });
        var output = _console.GetOutput();
        var deserialized = JsonSerializer.Deserialize<Jwt>(output);
 
        Assert.Equal(Environment.UserName, deserialized.Name);
        Assert.Equal(DevJwtsDefaults.Scheme, deserialized.Scheme);
        Assert.Equal($"http://localhost:23528, https://localhost:44395, https://localhost:5001, http://localhost:5000", deserialized.Audience);
    }
 
    [Fact]
    public void PrintCommand_ShowsCustomizedOptions()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--role", "foobar" });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
 
        app.Run(new[] { "print", id, "--project", project });
        var output = _console.GetOutput();
 
        Assert.Contains($"ID: {id}", output);
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains($"Scheme: {DevJwtsDefaults.Scheme}", output);
        Assert.Contains($"Audience(s): http://localhost:23528, https://localhost:44395, https://localhost:5001, http://localhost:5000", output);
        Assert.Contains($"Roles: [foobar]", output);
        Assert.DoesNotContain("Custom Claims", output);
    }
 
    [Fact]
    public void PrintComamnd_ShowsAllOptionsWithShowAll()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--claim", "foo=bar" });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
 
        app.Run(new[] { "print", id, "--project", project, "--show-all" });
        var output = _console.GetOutput();
 
        Assert.Contains($"ID: {id}", output);
        Assert.Contains($"Name: {Environment.UserName}", output);
        Assert.Contains($"Scheme: {DevJwtsDefaults.Scheme}", output);
        Assert.Contains($"Audience(s): http://localhost:23528, https://localhost:44395, https://localhost:5001, http://localhost:5000", output);
        Assert.Contains($"Scopes: none", output);
        Assert.Contains($"Roles: [none]", output);
        Assert.Contains($"Custom Claims: [foo=bar]", output);
    }
 
    [Fact]
    public void Create_WithJsonOutput_CanBeSerialized()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "--output", "json" });
        var output = _console.GetOutput();
        var deserialized = JsonSerializer.Deserialize<Jwt>(output);
 
        Assert.NotNull(deserialized);
        Assert.Equal(DevJwtsDefaults.Scheme, deserialized.Scheme);
        Assert.Equal(Environment.UserName, deserialized.Name);
    }
 
    [Fact]
    public void Create_WithTokenOutput_ProducesSingleValue()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project, "-o", "token" });
        var output = _console.GetOutput();
 
        var handler = new JwtSecurityTokenHandler();
        Assert.True(handler.CanReadToken(output.Trim()));
    }
 
    [Fact]
    public void Create_GracefullyHandles_NoLaunchSettings()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var app = new Program(_console);
        var launchSettingsPath = Path.Combine(projectPath, "Properties", "launchSettings.json");
 
        File.Delete(launchSettingsPath);
 
        app.Run(new[] { "create", "--project", project });
        var output = _console.GetOutput();
 
        Assert.Contains(Resources.CreateCommand_NoAudience_Error, output);
    }
 
    [Fact]
    public async Task Create_GracefullyHandles_PrepopulatedSecrets()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
        await File.WriteAllTextAsync(secretsFilePath,
@"{
  ""Foo"": {
    ""Bar"": ""baz""
  }
}"{
  ""Foo"": {
    ""Bar"": ""baz""
  }
}");
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project });
        var output = _console.GetOutput();
 
        Assert.Contains("New JWT saved", output);
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        Assert.NotNull(secretsJson);
        var signingKey = Assert.Single(secretsJson[SigningKeysHandler.GetSigningKeyPropertyName(DevJwtsDefaults.Scheme)].AsArray());
        Assert.Equal(32, signingKey["Length"].GetValue<int>());
        Assert.True(Convert.TryFromBase64String(signingKey["Value"].GetValue<string>(), new byte[32], out var _));
        Assert.True(secretsJson.TryGetPropertyValue("Foo", out var fooField));
        Assert.Equal("baz", fooField["Bar"].GetValue<string>());
    }
 
    [Fact]
    public async Task Create_GracefullyHandles_PrepopulatedSecrets_WithCommasAndComments()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
        await File.WriteAllTextAsync(secretsFilePath,
@"{
  ""Foo"": {
    ""Bar"": ""baz"",
    //""Bar"": ""baz"",
  }
}"{
  ""Foo"": {
    ""Bar"": ""baz"",
    //""Bar"": ""baz"",
  }
}");
        var app = new Program(_console);
        app.Run(["create", "--project", project]);
        var output = _console.GetOutput();
 
        Assert.Contains("New JWT saved", output);
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        Assert.NotNull(secretsJson);
        var signingKey = Assert.Single(secretsJson[SigningKeysHandler.GetSigningKeyPropertyName(DevJwtsDefaults.Scheme)].AsArray());
        Assert.Equal(32, signingKey["Length"].GetValue<int>());
        Assert.True(Convert.TryFromBase64String(signingKey["Value"].GetValue<string>(), new byte[32], out var _));
        Assert.True(secretsJson.TryGetPropertyValue("Foo", out var fooField));
        Assert.Equal("baz", fooField["Bar"].GetValue<string>());
    }
 
    [Fact]
    public void Create_GetsAudiencesFromAllIISAndKestrel()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project });
        var matches = Regex.Matches(_console.GetOutput(), "New JWT saved with ID '(.*?)'");
        var id = matches.SingleOrDefault().Groups[1].Value;
        app.Run(new[] { "print", id, "--project", project, "--show-all" });
        var output = _console.GetOutput();
 
        Assert.Contains("New JWT saved", output);
        Assert.Contains($"Audience(s): http://localhost:23528, https://localhost:44395, https://localhost:5001, http://localhost:5000", output);
    }
 
    [Fact]
    public async Task Create_SupportsSettingACustomIssuerAndScheme()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        Assert.True(secretsJson.ContainsKey(SigningKeysHandler.GetSigningKeyPropertyName("test-scheme")));
        var signingKey = Assert.Single(secretsJson[SigningKeysHandler.GetSigningKeyPropertyName("test-scheme")].AsArray());
        Assert.Equal(32, signingKey["Length"].GetValue<int>());
        Assert.True(Convert.TryFromBase64String(signingKey["Value"].GetValue<string>(), new byte[32], out var _));
        Assert.Equal("test-issuer", signingKey["Issuer"].GetValue<string>());
    }
 
    [Fact]
    public async Task Create_SupportsSettingMutlipleIssuersAndSingleScheme()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer-2", "--scheme", "test-scheme" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        Assert.True(secretsJson.ContainsKey(SigningKeysHandler.GetSigningKeyPropertyName("test-scheme")));
        var signingKeys = secretsJson[SigningKeysHandler.GetSigningKeyPropertyName("test-scheme")].AsArray();
        Assert.Equal(2, signingKeys.Count);
        Assert.NotNull(signingKeys.SingleOrDefault(signingKey => signingKey["Issuer"].GetValue<string>() == "test-issuer"));
        Assert.NotNull(signingKeys.SingleOrDefault(signingKey => signingKey["Issuer"].GetValue<string>() == "test-issuer-2"));
    }
 
    [Fact]
    public async Task Create_SupportsSettingSingleIssuerAndMultipleSchemes()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
        var secretsFilePath = PathHelper.GetSecretsPathFromSecretsId(fixture.TestSecretsId);
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme-2" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        using var openStream = File.OpenRead(secretsFilePath);
        var secretsJson = await JsonSerializer.DeserializeAsync<JsonObject>(openStream);
        var signingKey1 = Assert.Single(secretsJson[SigningKeysHandler.GetSigningKeyPropertyName("test-scheme")].AsArray());
        Assert.Equal("test-issuer", signingKey1["Issuer"].GetValue<string>());
        Assert.Equal(32, signingKey1["Length"].GetValue<int>());
        Assert.True(Convert.TryFromBase64String(signingKey1["Value"].GetValue<string>(), new byte[32], out var _));
        var signingKey2 = Assert.Single(secretsJson[SigningKeysHandler.GetSigningKeyPropertyName("test-scheme-2")].AsArray());
        Assert.Equal("test-issuer", signingKey2["Issuer"].GetValue<string>());
        Assert.Equal(32, signingKey2["Length"].GetValue<int>());
        Assert.True(Convert.TryFromBase64String(signingKey2["Value"].GetValue<string>(), new byte[32], out var _));
    }
 
    [Fact]
    public void Key_CanPrintAndReset_BySchemeAndIssuer()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme-2" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer-2", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer-2", "--scheme", "test-scheme-3" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
        _console.ClearOutput();
 
        app.Run(new[] { "key", "--project", project, "--scheme", "test-scheme", "--issuer", "test-issuer" });
        var printMatches = Regex.Matches(_console.GetOutput(), "Signing Key: '(.*?)'");
        var key = printMatches.SingleOrDefault().Groups[1].Value;
        _console.ClearOutput();
 
        app.Run(new[] { "key", "--project", project, "--reset", "--force", "--scheme", "test-scheme", "--issuer", "test-issuer" });
        var resetMatches = Regex.Matches(_console.GetOutput(), "New signing key created: '(.*?)'");
        var resetKey = resetMatches.SingleOrDefault().Groups[1].Value;
        Assert.NotEqual(key, resetKey);
    }
 
    [Fact]
    public void Key_CanPrintWithBase64()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer", "--scheme", "test-scheme-2" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer-2", "--scheme", "test-scheme" });
        app.Run(new[] { "create", "--project", project, "--issuer", "test-issuer-2", "--scheme", "test-scheme-3" });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
        _console.ClearOutput();
 
        app.Run(new[] { "key", "--project", project, "--scheme", "test-scheme", "--issuer", "test-issuer" });
        var printMatches = Regex.Matches(_console.GetOutput(), "Signing Key: '(.*?)'");
        var key = printMatches.SingleOrDefault().Groups[1].Value;
        _console.ClearOutput();
 
        var buffer = new Span<byte>(new byte[key.Length]);
        Assert.True(Convert.TryFromBase64String(key, buffer, out var bytesParsed));
        Assert.Equal(32, bytesParsed);
    }
 
    [Fact]
    public void Create_CanHandleNoProjectOptionProvided()
    {
        var projectPath = fixture.CreateProject();
        Directory.SetCurrentDirectory(projectPath);
 
        var app = new Program(_console);
        app.Run(["create"]);
 
        Assert.DoesNotContain("No project found at `-p|--project` path or current directory.", _console.GetOutput());
        Assert.Contains("New JWT saved", _console.GetOutput());
    }
 
    [Fact]
    public void Create_CanHandleNoProjectOptionProvided_WithNoProjects()
    {
        var path = Directory.CreateDirectory(Path.Combine(Path.GetTempPath(), "userjwtstest"));
        Directory.SetCurrentDirectory(path.FullName);
 
        var app = new Program(_console);
        app.Run(["create"]);
 
        Assert.Contains($"Could not find a MSBuild project file in '{Directory.GetCurrentDirectory()}'. Specify which project to use with the --project option.", _console.GetOutput());
        Assert.DoesNotContain(Resources.CreateCommand_NoAudience_Error, _console.GetOutput());
    }
 
    [Fact]
    public void Create_CanHandleAppsettingsOption_WithNoFile()
    {
        var projectPath = fixture.CreateProject();
        Directory.SetCurrentDirectory(projectPath);
        var expectedAppsettingsPath = Path.Combine(Directory.GetCurrentDirectory(), "appsettings.DoesNotExist.json");
 
        var app = new Program(_console);
        app.Run(["create", "--appsettings-file", "appsettings.DoesNotExist.json"]);
 
        Assert.Contains($"Could not find Appsettings file '{expectedAppsettingsPath}'. Check the filename and that the file exists.", _console.GetOutput());
    }
 
    [Fact]
    public void Delete_CanHandleNoProjectOptionProvided_WithNoProjects()
    {
        var path = Directory.CreateDirectory(Path.Combine(Path.GetTempPath(), "userjwtstest"));
        Directory.SetCurrentDirectory(path.FullName);
 
        var app = new Program(_console);
        app.Run(["remove", "some-id"]);
 
        Assert.Contains($"Could not find a MSBuild project file in '{Directory.GetCurrentDirectory()}'. Specify which project to use with the --project option.", _console.GetOutput());
    }
 
    [Fact]
    public void Delete_CanHandleAppsettingsOption_WithNoFile()
    {
        var projectPath = fixture.CreateProject();
        Directory.SetCurrentDirectory(projectPath);
        var expectedAppsettingsPath = Path.Combine(Directory.GetCurrentDirectory(), "appsettings.DoesNotExist.json");
 
        var app = new Program(_console);
        app.Run(["remove", "some-id", "--appsettings-file", "appsettings.DoesNotExist.json"]);
 
        Assert.Contains($"Could not find Appsettings file '{expectedAppsettingsPath}'. Check the filename and that the file exists.", _console.GetOutput());
    }
 
    [Fact]
    public void Clear_CanHandleNoProjectOptionProvided_WithNoProjects()
    {
        var path = Directory.CreateDirectory(Path.Combine(Path.GetTempPath(), "userjwtstest"));
        Directory.SetCurrentDirectory(path.FullName);
 
        var app = new Program(_console);
        app.Run(["clear"]);
 
        Assert.Contains($"Could not find a MSBuild project file in '{Directory.GetCurrentDirectory()}'. Specify which project to use with the --project option.", _console.GetOutput());
    }
 
    [Fact]
    public void Clear_CanHandleAppsettingsOption_WithNoFile()
    {
        var projectPath = fixture.CreateProject();
        Directory.SetCurrentDirectory(projectPath);
        var expectedAppsettingsPath = Path.Combine(Directory.GetCurrentDirectory(), "appsettings.DoesNotExist.json");
 
        var app = new Program(_console);
        app.Run(["clear", "--appsettings-file", "appsettings.DoesNotExist.json"]);
 
        Assert.Contains($"Could not find Appsettings file '{expectedAppsettingsPath}'. Check the filename and that the file exists.", _console.GetOutput());
    }
 
    [Fact]
    public void List_CanHandleNoProjectOptionProvided_WithNoProjects()
    {
        var path = Directory.CreateDirectory(Path.Combine(Path.GetTempPath(), "userjwtstest"));
        Directory.SetCurrentDirectory(path.FullName);
 
        var app = new Program(_console);
        app.Run(["list"]);
 
        Assert.Contains($"Could not find a MSBuild project file in '{Directory.GetCurrentDirectory()}'. Specify which project to use with the --project option.", _console.GetOutput());
    }
 
    [Fact]
    public void List_CanHandleProjectOptionAsPath()
    {
        var projectPath = fixture.CreateProject();
        var project = Path.Combine(projectPath, "TestProject.csproj");
 
        var app = new Program(_console);
        app.Run(new[] { "list", "--project", projectPath });
 
        Assert.Contains(Path.Combine(projectPath, project), _console.GetOutput());
    }
 
    [Fact]
    public void List_CanHandleRelativePathAsOption()
    {
        var projectPath = fixture.CreateProject();
        var tempPath = Path.GetTempPath();
        var targetPath = Path.GetRelativePath(tempPath, projectPath);
        var project = Path.Combine(projectPath, "TestProject.csproj");
        Directory.SetCurrentDirectory(tempPath);
 
        var app = new Program(_console);
        app.Run(new[] { "list", "--project", targetPath });
 
        Assert.DoesNotContain($"The project file '{targetPath}' does not exist.", _console.GetOutput());
        Assert.Contains(Path.Combine(projectPath, project), _console.GetOutput());
    }
 
    [Fact]
    public void Create_CanHandleRelativePathAsOption()
    {
        var projectPath = fixture.CreateProject();
        var tempPath = Path.GetTempPath();
        var targetPath = Path.GetRelativePath(tempPath, projectPath);
        Directory.SetCurrentDirectory(tempPath);
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", targetPath });
 
        Assert.DoesNotContain($"The project file '{targetPath}' does not exist.", _console.GetOutput());
        Assert.Contains("New JWT saved", _console.GetOutput());
    }
 
    [Fact]
    public void Create_CanHandleRelativePathAsOptionForAppsettingsOption()
    {
        var projectPath = fixture.CreateProject();
        var tempPath = Path.GetTempPath();
        var targetPath = Path.GetRelativePath(tempPath, projectPath);
        Directory.SetCurrentDirectory(tempPath);
 
        var app = new Program(_console);
        app.Run(new[] { "create", "--project", targetPath, "--appsettings-file", "appsettings.Local.json" });
        
        Assert.DoesNotContain($"Could not find Appsettings file '{projectPath}'. Check the filename and that the file exists.", _console.GetOutput());
        Assert.Contains("New JWT saved", _console.GetOutput());
    }
 
    [ConditionalFact]
    [OSSkipCondition(OperatingSystems.Windows, SkipReason = "UnixFileMode is not supported on Windows.")]
    public void Create_CreatesFileWithUserOnlyUnixFileMode()
    {
        var project = Path.Combine(fixture.CreateProject(), "TestProject.csproj");
        var app = new Program(_console);
 
        app.Run(new[] { "create", "--project", project });
 
        Assert.Contains("New JWT saved", _console.GetOutput());
 
        Assert.NotNull(app.UserJwtsFilePath);
        Assert.Equal(UnixFileMode.UserRead | UnixFileMode.UserWrite, File.GetUnixFileMode(app.UserJwtsFilePath));
    }
}