From 04adadaacfa044df6b0c556c9e849ce9232e8950 Mon Sep 17 00:00:00 2001 From: Oleksii Holub <1935960+Tyrrrz@users.noreply.github.com> Date: Tue, 6 Aug 2024 17:51:32 +0300 Subject: [PATCH] Add `Passwordless.AspNetCore` registration overloads that support `IConfiguration` properly (#146) --- .../IdentityBuilderExtensions.cs | 202 +++++++++++------- .../ServiceCollectionExtensions.cs | 68 +++--- .../Passwordless.Tests.csproj | 1 + .../ServiceRegistrationTests.cs | 80 +++++++ 4 files changed, 239 insertions(+), 112 deletions(-) diff --git a/src/Passwordless.AspNetCore/IdentityBuilderExtensions.cs b/src/Passwordless.AspNetCore/IdentityBuilderExtensions.cs index 3c8b237..4fe083e 100644 --- a/src/Passwordless.AspNetCore/IdentityBuilderExtensions.cs +++ b/src/Passwordless.AspNetCore/IdentityBuilderExtensions.cs @@ -18,47 +18,53 @@ namespace Microsoft.Extensions.DependencyInjection; /// public static class IdentityBuilderExtensions { - /// - /// Adds the services to support - /// - /// The . - /// Configures the . - /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IServiceCollection AddPasswordless(this IServiceCollection services, Action configure) - where TUser : class, new() + private static IServiceCollection AddPasswordlessIdentity( + this IServiceCollection services, + [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] + Type userType, + Action> configureOptions, + string? defaultScheme) { - return services.AddPasswordlessCore(typeof(TUser), configure, defaultScheme: null); + // Options + var optionsBuilder = services.AddOptions(); + configureOptions(optionsBuilder); + + // Default scheme + if (!string.IsNullOrEmpty(defaultScheme)) + { + optionsBuilder.Configure(o => o.SignInScheme = defaultScheme); + } + + // Services + services.TryAddScoped( + typeof(IPasswordlessService), + typeof(PasswordlessService<>).MakeGenericType(userType) + ); + + services.TryAddScoped(); + + return services; } /// /// Adds the services to support /// - /// The current instance. + /// The . /// Configures the . - /// The . + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IdentityBuilder AddPasswordless(this IdentityBuilder builder, Action configure) - { - builder.Services.AddPasswordlessCore(builder.UserType, configure, IdentityConstants.ApplicationScheme); - return builder; - } - - [RequiresDynamicCode("Calls Microsoft.Extensions.DependencyInjection.IdentityBuilderExtensions.AddShared(Type, OptionsBuilder, String)")] - [RequiresUnreferencedCode("Calls Microsoft.Extensions.DependencyInjection.IdentityBuilderExtensions.AddShared(Type, OptionsBuilder, String)")] - private static IServiceCollection AddPasswordlessCore(this IServiceCollection services, - [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type userType, - Action configure, - string? defaultScheme) + public static IServiceCollection AddPasswordless( + this IServiceCollection services, + Action configure) + where TUser : class, new() { - var optionsBuilder = services.AddOptions().Configure(configure); - - // Add the SDK services but don't configure it there since ASP.NET Core options are a superset of their options. + // Don't set up options here because we can't use the provided delegate as it's for a different type services.AddPasswordlessSdk(_ => { }); - // Override SDK options to come from ASP.NET Core options + // Derive core options from ASP.NET Core options services.AddOptions() .Configure>((options, aspNetCoreOptionsAccessor) => { @@ -68,106 +74,152 @@ private static IServiceCollection AddPasswordlessCore(this IServiceCollection se options.ApiKey = aspNetCoreOptions.ApiKey; }); - return services.AddShared(userType, optionsBuilder, defaultScheme); + return services.AddPasswordlessIdentity(typeof(TUser), o => o.Configure(configure), null); } /// /// Adds the services to support /// - /// The current instance. - /// The to use to bind to . Generally it's own section. - /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IServiceCollection AddPasswordless(this IdentityBuilder builder, IConfiguration configuration) + public static IServiceCollection AddPasswordless( + this IServiceCollection services, + IConfiguration configuration) + where TUser : class, new() { - return builder.AddPasswordless(configuration, builder.UserType, IdentityConstants.ApplicationScheme); + services.AddPasswordlessSdk(configuration); + return services.AddPasswordlessIdentity(typeof(TUser), o => o.Bind(configuration), null); } /// /// Adds the services to support /// - /// The current instance. - /// The to use to bind to . Generally it's own section. - /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IServiceCollection AddPasswordless(this IdentityBuilder builder, IConfiguration configuration) + public static IServiceCollection AddPasswordless( + this IServiceCollection services, + string configurationSection) + where TUser : class, new() { - return builder.AddPasswordless(configuration, typeof(TUserType), null); + services.AddPasswordlessSdk(configurationSection); + return services.AddPasswordlessIdentity(typeof(TUser), o => o.BindConfiguration(configurationSection), null); } + /// + /// Adds the services to support + /// + /// The current instance. + /// Configures the . + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - private static IServiceCollection AddPasswordless(this IdentityBuilder builder, IConfiguration configuration, Type? userType, string? defaultScheme) + public static IdentityBuilder AddPasswordless( + this IdentityBuilder identity, + Action configure) { - var optionsBuilder = builder.Services - .AddOptions() - .Bind(configuration); + // Don't set up options here because we can't use the provided delegate as it's for a different type + identity.Services.AddPasswordlessSdk(_ => { }); + + // Derive core options from ASP.NET Core options + identity.Services.AddOptions() + .Configure>((options, aspNetCoreOptionsAccessor) => + { + var aspNetCoreOptions = aspNetCoreOptionsAccessor.Value; + options.ApiUrl = aspNetCoreOptions.ApiUrl; + options.ApiSecret = aspNetCoreOptions.ApiSecret; + options.ApiKey = aspNetCoreOptions.ApiKey; + }); - builder.Services.AddPasswordlessSdk(configuration); + identity.Services.AddPasswordlessIdentity( + identity.UserType, + o => o.Configure(configure), + IdentityConstants.ApplicationScheme + ); - return builder.Services.AddShared(userType ?? builder.UserType, optionsBuilder, IdentityConstants.ApplicationScheme); + return identity; } /// /// Adds the services to support /// - /// The current instance. - /// The configuration path to use to bind to . - /// The . + /// The current instance. + /// The to use to bind to . Generally it's own section. + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IServiceCollection AddPasswordless(this IdentityBuilder builder, string path) + public static IdentityBuilder AddPasswordless(this IdentityBuilder identity, IConfiguration configuration) { - return builder.AddPasswordless(path, builder.UserType, IdentityConstants.ApplicationScheme); + identity.Services.AddPasswordlessSdk(configuration); + + identity.Services.AddPasswordlessIdentity( + identity.UserType, + o => o.Bind(configuration), + IdentityConstants.ApplicationScheme + ); + + return identity; } /// /// Adds the services to support /// - /// The current instance. - /// The configuration path to use to bind to . - /// The . + /// The current instance. + /// The to use to bind to . Generally it's own section. + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - public static IServiceCollection AddPasswordless(this IdentityBuilder builder, string path) + public static IdentityBuilder AddPasswordless(this IdentityBuilder identity, IConfiguration configuration) { - return builder.AddPasswordless(path, typeof(TUserType), null); + identity.Services.AddPasswordlessSdk(configuration); + + identity.Services.AddPasswordlessIdentity( + typeof(TUserType), + o => o.Bind(configuration), + IdentityConstants.ApplicationScheme + ); + + return identity; } + /// + /// Adds the services to support + /// + /// The current instance. + /// The configuration path to use to bind to . + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - private static IServiceCollection AddPasswordless(this IdentityBuilder builder, string path, Type userType, string? defaultScheme) + public static IdentityBuilder AddPasswordless(this IdentityBuilder identity, string configurationSection) { - var optionsBuilder = builder.Services - .AddOptions() - .BindConfiguration(path); + identity.Services.AddPasswordlessSdk(configurationSection); - builder.Services.AddPasswordlessSdk(path); + identity.Services.AddPasswordlessIdentity( + identity.UserType, + o => o.BindConfiguration(configurationSection), + IdentityConstants.ApplicationScheme + ); - return builder.Services.AddShared(userType ?? builder.UserType, optionsBuilder, IdentityConstants.ApplicationScheme); + return identity; } + /// + /// Adds the services to support + /// + /// The current instance. + /// The configuration path to use to bind to . + /// The . [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] - private static IServiceCollection AddShared(this IServiceCollection services, - [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] - Type userType, - OptionsBuilder optionsBuilder, - string? defaultScheme) + public static IdentityBuilder AddPasswordless(this IdentityBuilder identity, string configurationSection) { - if (!string.IsNullOrEmpty(defaultScheme)) - { - optionsBuilder.Configure(o => o.SignInScheme = defaultScheme); - } + identity.Services.AddPasswordlessSdk(configurationSection); - services.TryAddScoped( - typeof(IPasswordlessService), - typeof(PasswordlessService<>).MakeGenericType(userType)); + identity.Services.AddPasswordlessIdentity( + typeof(TUserType), + o => o.BindConfiguration(configurationSection), + IdentityConstants.ApplicationScheme + ); - services.TryAddScoped(); - - return services; + return identity; } } \ No newline at end of file diff --git a/src/Passwordless/ServiceCollectionExtensions.cs b/src/Passwordless/ServiceCollectionExtensions.cs index 8fd2164..8cedc7e 100644 --- a/src/Passwordless/ServiceCollectionExtensions.cs +++ b/src/Passwordless/ServiceCollectionExtensions.cs @@ -1,5 +1,4 @@ using System; -using System.Diagnostics.CodeAnalysis; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.Options; using Passwordless; @@ -12,67 +11,62 @@ namespace Microsoft.Extensions.DependencyInjection; /// public static class ServiceCollectionExtensions { - /// - /// Adds and configures Passwordless-related services. - /// - public static IServiceCollection AddPasswordlessSdk( + private static IServiceCollection AddPasswordlessSdk( this IServiceCollection services, - Action configureOptions) + Action> configureOptions) { - services.AddOptions().Configure(configureOptions); - services.RegisterDependencies(); + // Options + configureOptions( + services.AddOptions() + ); + + // Client + services.AddHttpClient((http, sp) => + new PasswordlessClient(http, sp.GetRequiredService>().Value) + ); + + // TODO: Get rid of this service, all consumers should use the interface + services.AddTransient(sp => (PasswordlessClient)sp.GetRequiredService()); return services; } + /// + /// Adds and configures Passwordless-related services. + /// + public static IServiceCollection AddPasswordlessSdk( + this IServiceCollection services, + Action configureOptions) => + services.AddPasswordlessSdk(o => o.Configure(configureOptions)); + /// /// Adds and configures Passwordless-related services. /// #if NET6_0_OR_GREATER - [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] + [System.Diagnostics.CodeAnalysis.RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] #endif #if NET7_0_OR_GREATER - [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] + [System.Diagnostics.CodeAnalysis.RequiresDynamicCode("This method is incompatible with native AOT compilation.")] #endif public static IServiceCollection AddPasswordlessSdk( this IServiceCollection services, - IConfiguration configuration) - { - services.AddOptions().Configure(configuration.Bind); - services.RegisterDependencies(); - - return services; - } + IConfiguration configuration) => + services.AddPasswordlessSdk(o => o.Bind(configuration)); /// /// Adds and configures Passwordless-related services. /// /// - /// + /// /// #if NET6_0_OR_GREATER - [RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] + [System.Diagnostics.CodeAnalysis.RequiresUnreferencedCode("This method is incompatible with assembly trimming.")] #endif #if NET7_0_OR_GREATER - [RequiresDynamicCode("This method is incompatible with native AOT compilation.")] + [System.Diagnostics.CodeAnalysis.RequiresDynamicCode("This method is incompatible with native AOT compilation.")] #endif public static IServiceCollection AddPasswordlessSdk( this IServiceCollection services, - string section) - { - services.AddOptions().BindConfiguration(section); - services.RegisterDependencies(); - - return services; - } - - private static void RegisterDependencies(this IServiceCollection services) - { - services.AddHttpClient((http, sp) => - new PasswordlessClient(http, sp.GetRequiredService>().Value) - ); - - // TODO: Get rid of this service, all consumers should use the interface - services.AddTransient(sp => (PasswordlessClient)sp.GetRequiredService()); - } + string configurationSection) => + services.AddPasswordlessSdk(o => o.BindConfiguration(configurationSection)); } \ No newline at end of file diff --git a/tests/Passwordless.Tests/Passwordless.Tests.csproj b/tests/Passwordless.Tests/Passwordless.Tests.csproj index d95da88..b120d81 100644 --- a/tests/Passwordless.Tests/Passwordless.Tests.csproj +++ b/tests/Passwordless.Tests/Passwordless.Tests.csproj @@ -16,6 +16,7 @@ + diff --git a/tests/Passwordless.Tests/ServiceRegistrationTests.cs b/tests/Passwordless.Tests/ServiceRegistrationTests.cs index f93d7ad..8cdb179 100644 --- a/tests/Passwordless.Tests/ServiceRegistrationTests.cs +++ b/tests/Passwordless.Tests/ServiceRegistrationTests.cs @@ -1,12 +1,69 @@ using System; +using System.Collections.Generic; using System.Threading.Tasks; +using FluentAssertions; +using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Options; using Xunit; namespace Passwordless.Tests; public class ServiceRegistrationTests { + [Fact] + public void I_can_register_a_client_with_options_derived_from_configuration() + { + // Arrange + var configuration = new ConfigurationBuilder() + .AddInMemoryCollection([ + new KeyValuePair("Passwordless:ApiSecret", "foo"), + new KeyValuePair("Passwordless:ApiKey", "bar") + ]) + .Build(); + + var services = new ServiceCollection() + .AddSingleton(configuration) + .AddPasswordlessSdk(configuration.GetSection("Passwordless")) + .BuildServiceProvider(); + + // Act + var options = services.GetRequiredService>(); + + // Assert + options.Value.ApiSecret.Should().Be("foo"); + options.Value.ApiKey.Should().Be("bar"); + } + + [Fact] + public void I_can_register_a_client_and_update_its_options_at_any_point_in_time() + { + // Arrange + var configuration = new ConfigurationBuilder() + .AddInMemoryCollection([ + new KeyValuePair("Passwordless:ApiSecret", "foo"), + new KeyValuePair("Passwordless:ApiKey", "bar") + ]) + .Build(); + + var services = new ServiceCollection() + .AddSingleton(configuration) + .AddPasswordlessSdk(configuration.GetSection("Passwordless")) + .BuildServiceProvider(); + + // Act + foreach (var provider in configuration.Providers) + { + provider.Set("Passwordless:ApiSecret", "baz"); + provider.Set("Passwordless:ApiKey", "zap"); + } + + // Assert + var options = services.GetRequiredService>(); + options.Value.ApiSecret.Should().Be("baz"); + options.Value.ApiKey.Should().Be("zap"); + } + [Fact] public async Task I_can_register_a_client_with_invalid_credentials_and_not_receive_an_error_until_it_is_used() { @@ -26,4 +83,27 @@ await Assert.ThrowsAnyAsync(async () => await passwordless.CreateRegisterTokenAsync(new RegisterOptions("user123", "User")) ); } + + [Fact] + public void I_can_register_a_client_and_retrieve_a_new_copy_for_each_scope() + { + // Arrange + var services = new ServiceCollection() + .AddPasswordlessSdk(o => + { + o.ApiSecret = "foo"; + o.ApiKey = "bar"; + }) + .BuildServiceProvider(); + + // Act + using var scope1 = services.CreateScope(); + var client1 = scope1.ServiceProvider.GetRequiredService(); + + using var scope2 = services.CreateScope(); + var client2 = scope2.ServiceProvider.GetRequiredService(); + + // Assert + client1.Should().NotBeSameAs(client2); + } } \ No newline at end of file