From 84b18e9de7757f2fcaa45f6005f6a0a993765a9d Mon Sep 17 00:00:00 2001 From: Justin Baur <19896123+justindbaur@users.noreply.github.com> Date: Fri, 28 Jun 2024 10:28:07 -0400 Subject: [PATCH] Add overriddable configuration (#4437) * Add Overridable Configuration * Add Remarks --- .../Factories/ApiApplicationFactory.cs | 4 +- .../Factories/WebApplicationFactoryBase.cs | 71 ++++++++++++++++--- 2 files changed, 65 insertions(+), 10 deletions(-) diff --git a/test/Api.IntegrationTest/Factories/ApiApplicationFactory.cs b/test/Api.IntegrationTest/Factories/ApiApplicationFactory.cs index 3938ef46b3..230f0bcf08 100644 --- a/test/Api.IntegrationTest/Factories/ApiApplicationFactory.cs +++ b/test/Api.IntegrationTest/Factories/ApiApplicationFactory.cs @@ -4,6 +4,8 @@ using Microsoft.AspNetCore.Authentication.JwtBearer; using Microsoft.AspNetCore.TestHost; using Microsoft.Data.Sqlite; +#nullable enable + namespace Bit.Api.IntegrationTest.Factories; public class ApiApplicationFactory : WebApplicationFactoryBase @@ -62,7 +64,7 @@ public class ApiApplicationFactory : WebApplicationFactoryBase protected override void Dispose(bool disposing) { base.Dispose(disposing); - SqliteConnection.Dispose(); + SqliteConnection!.Dispose(); } /// diff --git a/test/IntegrationTestCommon/Factories/WebApplicationFactoryBase.cs b/test/IntegrationTestCommon/Factories/WebApplicationFactoryBase.cs index d1e4fc3065..62cb5c624a 100644 --- a/test/IntegrationTestCommon/Factories/WebApplicationFactoryBase.cs +++ b/test/IntegrationTestCommon/Factories/WebApplicationFactoryBase.cs @@ -16,6 +16,8 @@ using Microsoft.Extensions.Logging.Abstractions; using NSubstitute; using NoopRepos = Bit.Core.Repositories.Noop; +#nullable enable + namespace Bit.IntegrationTestCommon.Factories; public static class FactoryConstants @@ -32,9 +34,11 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory /// /// This will need to be set BEFORE using the Server property /// - public SqliteConnection SqliteConnection { get; set; } + public SqliteConnection? SqliteConnection { get; set; } private readonly List> _configureTestServices = new(); + private readonly List> _configureAppConfiguration = new(); + private bool _handleSqliteDisposal { get; set; } @@ -53,6 +57,50 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory }); } + /// + /// Add your own configuration provider to the application. + /// + /// The action adding your own providers. + /// This needs to be ran BEFORE making any calls through the factory to take effect. + /// + /// + /// factory.UpdateConfiguration(builder => + /// { + /// builder.AddInMemoryCollection(new Dictionary<string, string?> + /// { + /// { "globalSettings:attachment:connectionString", null}, + /// { "globalSettings:events:connectionString", null}, + /// }) + /// }) + /// + /// + public void UpdateConfiguration(Action configure) + { + _configureAppConfiguration.Add(configure); + } + + /// + /// Updates a single configuration entry for multiple entries at once use . + /// + /// The fully qualified name of the setting, using : as delimiter between sections. + /// The value of the setting. + /// This needs to be ran BEFORE making any calls through the factory to take effect. + /// + /// + /// factory.UpdateConfiguration("globalSettings:attachment:connectionString", null); + /// + /// + public void UpdateConfiguration(string key, string? value) + { + _configureAppConfiguration.Add(builder => + { + builder.AddInMemoryCollection(new Dictionary + { + { key, value }, + }); + }); + } + /// /// Configure the web host to use a SQLite in memory database /// @@ -73,7 +121,7 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory c.AddUserSecrets(typeof(Identity.Startup).Assembly, optional: true); - c.AddInMemoryCollection(new Dictionary + c.AddInMemoryCollection(new Dictionary { // Manually insert a EF provider so that ConfigureServices will add EF repositories but we will override // DbContextOptions to use an in memory database @@ -96,12 +144,16 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory // Email Verification { "globalSettings:enableEmailVerification", "true" }, - {"globalSettings:launchDarkly:flagValues:email-verification", "true" } - - + { "globalSettings:launchDarkly:flagValues:email-verification", "true" } }); }); + // Run configured actions after defaults to allow them to take precedence + foreach (var configureAppConfiguration in _configureAppConfiguration) + { + builder.ConfigureAppConfiguration(configureAppConfiguration); + } + builder.ConfigureTestServices(services => { var dbContextOptions = services.First(sd => sd.ServiceType == typeof(DbContextOptions)); @@ -193,10 +245,11 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory return scope.ServiceProvider.GetRequiredService(); } - public TS GetService() + public TService GetService() + where TService : notnull { var scope = Services.CreateScope(); - return scope.ServiceProvider.GetRequiredService(); + return scope.ServiceProvider.GetRequiredService(); } protected override void Dispose(bool disposing) @@ -204,7 +257,7 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory base.Dispose(disposing); if (_handleSqliteDisposal) { - SqliteConnection.Dispose(); + SqliteConnection!.Dispose(); } } @@ -213,7 +266,7 @@ public abstract class WebApplicationFactoryBase : WebApplicationFactory var serviceProvider = serviceCollection.BuildServiceProvider(); using var scope = serviceProvider.CreateScope(); var services = scope.ServiceProvider; - var context = services.GetService(); + var context = services.GetRequiredService(); if (_handleSqliteDisposal) { context.Database.EnsureDeleted();