using System; using System.Net.Http; using System.Threading.Tasks; using BuildingBlocks.Core.Model; using BuildingBlocks.EFCore; using BuildingBlocks.MassTransit; using BuildingBlocks.MessageProcessor; using BuildingBlocks.Mongo; using BuildingBlocks.Web; using DotNetCore.CAP.MongoDB; using Flight.Data; using FluentAssertions.Common; using Grpc.Net.Client; using MassTransit; using MassTransit.Testing; using MediatR; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc.Testing; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection.Extensions; using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Options; using Mongo2Go; using NSubstitute; using Respawn; using Serilog; using Xunit; using Xunit.Abstractions; namespace Integration.Test; [CollectionDefinition(nameof(IntegrationTestFixture))] public class FixtureCollection : ICollectionFixture { } public class IntegrationTestFixture : IAsyncLifetime { private WebApplicationFactory _factory; private MongoDbRunner _mongoRunner; public Checkpoint Checkpoint { get; set; } public Action? TestRegistrationServices { get; set; } public IServiceProvider ServiceProvider => _factory.Services; public IConfiguration Configuration => _factory.Services.GetRequiredService(); public HttpClient HttpClient => _factory.CreateClient(); public ITestHarness TestHarness => CreateHarness(); public GrpcChannel Channel => CreateChannel(); // ref: https://github.com/trbenning/serilog-sinks-xunit public ILogger CreateLogger(ITestOutputHelper output) { if (output != null) { return new LoggerConfiguration() .WriteTo.TestOutput(output) .CreateLogger(); } return null; } public void RegisterTestServices(Action services) => TestRegistrationServices = services; public virtual Task InitializeAsync() { _factory = new WebApplicationFactory() .WithWebHostBuilder(builder => { builder.UseEnvironment("test"); builder.ConfigureServices(services => { services.AddMassTransitTestHarness(x => { x.UsingRabbitMq((context, cfg) => { var rabbitMqOptions = services.GetOptions("RabbitMq"); var host = rabbitMqOptions.HostName; cfg.Host(host, h => { h.Username(rabbitMqOptions.UserName); h.Password(rabbitMqOptions.Password); }); cfg.ConfigureEndpoints(context); }); }); Checkpoint = new Checkpoint {TablesToIgnore = new[] {"__EFMigrationsHistory"}}; }); }); _mongoRunner = MongoDbRunner.Start(); var mongoOptions = _factory.Services.GetRequiredService>(); mongoOptions.Value.ConnectionString = _mongoRunner.ConnectionString; return Task.CompletedTask; } public virtual async Task DisposeAsync() { if (!string.IsNullOrEmpty(Configuration?.GetConnectionString("DefaultConnection"))) await Checkpoint.Reset(Configuration?.GetConnectionString("DefaultConnection")); await _factory.DisposeAsync(); _mongoRunner.Dispose(); } public async Task ExecuteScopeAsync(Func action) { using var scope = ServiceProvider.CreateScope(); await action(scope.ServiceProvider); } public async Task ExecuteScopeAsync(Func> action) { using var scope = ServiceProvider.CreateScope(); var result = await action(scope.ServiceProvider); return result; } public Task ExecuteDbContextAsync(Func action) { return ExecuteScopeAsync(sp => action(sp.GetService())); } public Task ExecuteDbContextAsync(Func action) { return ExecuteScopeAsync(sp => action(sp.GetService()).AsTask()); } public Task ExecuteDbContextAsync(Func action) { return ExecuteScopeAsync(sp => action(sp.GetService(), sp.GetService())); } public Task ExecuteDbContextAsync(Func> action) { return ExecuteScopeAsync(sp => action(sp.GetService())); } public Task ExecuteDbContextAsync(Func> action) { return ExecuteScopeAsync(sp => action(sp.GetService()).AsTask()); } public Task ExecuteDbContextAsync(Func> action) { return ExecuteScopeAsync(sp => action(sp.GetService(), sp.GetService())); } public Task InsertAsync(params T[] entities) where T : class { return ExecuteDbContextAsync(db => { foreach (var entity in entities) db.Set().Add(entity); return db.SaveChangesAsync(); }); } public Task InsertAsync(TEntity entity) where TEntity : class { return ExecuteDbContextAsync(db => { db.Set().Add(entity); return db.SaveChangesAsync(); }); } public Task InsertAsync(TEntity entity, TEntity2 entity2) where TEntity : class where TEntity2 : class { return ExecuteDbContextAsync(db => { db.Set().Add(entity); db.Set().Add(entity2); return db.SaveChangesAsync(); }); } public Task InsertAsync(TEntity entity, TEntity2 entity2, TEntity3 entity3) where TEntity : class where TEntity2 : class where TEntity3 : class { return ExecuteDbContextAsync(db => { db.Set().Add(entity); db.Set().Add(entity2); db.Set().Add(entity3); return db.SaveChangesAsync(); }); } public Task InsertAsync(TEntity entity, TEntity2 entity2, TEntity3 entity3, TEntity4 entity4) where TEntity : class where TEntity2 : class where TEntity3 : class where TEntity4 : class { return ExecuteDbContextAsync(db => { db.Set().Add(entity); db.Set().Add(entity2); db.Set().Add(entity3); db.Set().Add(entity4); return db.SaveChangesAsync(); }); } public Task FindAsync(long id) where T : class, IEntity { return ExecuteDbContextAsync(db => db.Set().FindAsync(id).AsTask()); } public Task SendAsync(IRequest request) { return ExecuteScopeAsync(sp => { var mediator = sp.GetRequiredService(); return mediator.Send(request); }); } public Task SendAsync(IRequest request) { return ExecuteScopeAsync(sp => { var mediator = sp.GetRequiredService(); return mediator.Send(request); }); } private ITestHarness CreateHarness() { var harness = ServiceProvider.GetTestHarness(); harness.Start().GetAwaiter().GetResult(); return harness; } private GrpcChannel CreateChannel() { return GrpcChannel.ForAddress(HttpClient.BaseAddress!, new GrpcChannelOptions {HttpClient = HttpClient}); } private IHttpContextAccessor AddHttpContextAccessorMock(IServiceProvider serviceProvider) { var httpContextAccessorMock = Substitute.For(); using var scope = serviceProvider.CreateScope(); httpContextAccessorMock.HttpContext = new DefaultHttpContext {RequestServices = scope.ServiceProvider}; httpContextAccessorMock.HttpContext.Request.Host = new HostString("localhost", 6012); httpContextAccessorMock.HttpContext.Request.Scheme = "http"; return httpContextAccessorMock; } }