我已经基于https://github.com/jasontaylordev/CleanArchitecture建立了一个项目。但是我在为控制器编写集成测试时遇到了一些麻烦,因为内存数据库不会在每个测试之间重置。每个测试都使用WebApplicationFactory来设置测试Web服务器,
public class CustomWebApplicationFactory<TStartup> : WebApplicationFactory<TStartup> where TStartup : class
{
protected override void ConfigureWebHost(IWebHostBuilder builder)
{
builder.ConfigureServices(services =>
{
// Remove the app's ApplicationDbContext registration.
var descriptor = services.SingleOrDefault(
d => d.ServiceType ==
typeof(DbContextOptions<ApplicationDbContext>));
if (descriptor != null)
{
services.Remove(descriptor);
}
// Add a database context using an in-memory
// database for testing.
services.AddDbContext<ApplicationDbContext>(options =>
{
options.UseInMemoryDatabase("InMemoryDbForTesting");
});
// Register test services
services.AddScoped<ICurrentUserService, TestCurrentUserService>();
services.AddScoped<IDateTime, TestDateTimeService>();
services.AddScoped<IIdentityService, TestIdentityService>();
// Build the service provider
var sp = services.BuildServiceProvider();
// Create a scope to obtain a reference to the database
// context (ApplicationDbContext).
using (var scope = sp.CreateScope())
{
var scopedServices = scope.ServiceProvider;
var context = scopedServices.GetRequiredService<ApplicationDbContext>();
var logger = scopedServices.GetRequiredService<ILogger<CustomWebApplicationFactory<TStartup>>>();
// Ensure the database is created.
context.Database.EnsureCreated();
try
{
// Seed the database with test data.
SeedSampleData(context);
}
catch (Exception ex)
{
logger.LogError(ex, "An error occurred seeding the database with test messages. Error: {Message}", ex.Message);
}
}
})
.UseEnvironment("Test");
}
...
以下创建测试的地方:
namespace CleanArchitecture.WebUI.IntegrationTests.Controllers.TodoItems
{
public class Create : IClassFixture<CustomWebApplicationFactory<Startup>>
{
private readonly CustomWebApplicationFactory<Startup> _factory;
public Create(CustomWebApplicationFactory<Startup> factory)
{
_factory = factory;
}
[Fact]
public async Task GivenValidCreateTodoItemCommand_ReturnsSuccessCode()
{
var client = await _factory.GetAuthenticatedClientAsync();
var command = new CreateTodoItemCommand
{
Title = "Do yet another thing."
};
var content = IntegrationTestHelper.GetRequestContent(command);
var response = await client.PostAsync($"/api/todoitems", content);
response.EnsureSuccessStatusCode();
}
...
以及以下阅读测试:
namespace CleanArchitecture.WebUI.IntegrationTests.Controllers.TodoItems
{
public class Read: IClassFixture<CustomWebApplicationFactory<Startup>>
{
private readonly CustomWebApplicationFactory<Startup> _factory;
public Read(CustomWebApplicationFactory<Startup> factory)
{
_factory = factory;
}
[Fact]
public async Task ShouldRetriveAllTodos()
{
var client = await _factory.GetAuthenticatedClientAsync();
var response = await client.GetAsync($"/api/todoitems");
var todos = Deserialize(response);
todos.Count().Should().Be(1); //but is 2 because database is shared between read and create test class.
}
问题在于两次测试之间未重置内存数据库。我尝试使用Guid.New.ToString
为内存数据库生成一个不同的名称,但是测试没有找到种子数据库数据,而是将测试放在相同的XUnit集合中,无济于事。
有什么好主意如何使测试不共享数据库?
答案 0 :(得分:1)
在创建DbContext时,我们通过以下方式将测试彼此隔离:
private static DbContextOptions<ApplicationDbContext> CreateDbContextOptions()
{
return new DbContextOptionsBuilder<ApplicationDbContext>()
//Database with same name gets reused, so let's isolate the tests from each other...
.UseInMemoryDatabase(Guid.NewGuid().ToString())
.Options;
}
在测试课中:
using (var context = new ApplicationDbContext(DbContextTestHelper.CreateDbContextOptions())
{
//arrange data in context
var testee = new XY(context);
//do test
//do asserts
}
答案 1 :(得分:0)
对我有用的是为每个WebApplicationFactory
实例生成一个DBName,然后为每个测试实例化其中一个实例。所以测试看起来像这样:
[Fact]
public void Test()
{
// Arrange
var appFactory = new WebApplicationFactory();
// ...
// Act
// ...
// Test
// ...
}
还有WebApplicationFactory
:
public class TestWebApplicationFactory : WebApplicationFactory<Startup>
{
private readonly string _dbName = Guid.NewGuid().ToString();
protected override void ConfigureWebHost(IWebHostBuilder builder)
{
base.ConfigureWebHost(builder);
builder.ConfigureServices(services =>
{
services.AddDbContext<DbContext>(options =>
{
// This is what makes a unique in-memory database per instance of TestWebApplicationFactory
options.UseInMemoryDatabase(_dbName);
});
});
}
}