网络工程师和网站开发员,浙江省建设局房管科网站,深圳软件开发公司在哪里,百度免费网站申请注册XUnit 依赖注入Intro现在的开发中越来越看重依赖注入的思想#xff0c;微软的 Asp.Net Core 框架更是天然集成了依赖注入#xff0c;那么在单元测试中如何使用依赖注入呢#xff1f;本文主要介绍如何通过 XUnit 来实现依赖注入#xff0c; XUnit 主要借助 SharedContext 来… XUnit 依赖注入Intro现在的开发中越来越看重依赖注入的思想微软的 Asp.Net Core 框架更是天然集成了依赖注入那么在单元测试中如何使用依赖注入呢本文主要介绍如何通过 XUnit 来实现依赖注入 XUnit 主要借助 SharedContext 来共享一部分资源包括这些资源的创建以及释放。Scoped针对 Scoped 的对象可以借助 XUnit 中的 IClassFixture 来实现定义自己的 Fixture需要初始化的资源在构造方法里初始化如果需要在测试结束的时候释放资源需要实现 IDisposable 接口需要依赖注入的测试类实现接口 IClassFixtureFixture在构造方法中注入实现的 Fixture 对象并在构造方法中使用 Fixture 对象中暴露的公共成员Singleton针对 Singleton 的对象可以借助 XUnit 中的 ICollectionFixture 来实现定义自己的 Fixture需要初始化的资源在构造方法里初始化如果需要在测试结束的时候释放资源需要实现 IDisposable 接口创建 CollectionDefinition实现接口 ICollectionFixtureFixture并添加一个 [CollectionDefinition(CollectionName)] Attribute CollectionName 需要在整个测试中唯一不能出现重复的 CollectionName在需要注入的测试类中添加 [Collection(CollectionName)] Attribute然后在构造方法中注入对应的 FixtureTips如果有多个类需要依赖注入可以通过一个基类来做这样就只需要一个基类上添加 [Collection(CollectionName)] Attribute其他类只需要集成这个基类就可以了SamplesScoped Sample这里直接以 XUnit 的示例为例public class DatabaseFixture : IDisposable
{ public DatabaseFixture() { Db new SqlConnection(MyConnectionString); // ... initialize data in the test database ... } public void Dispose() { // ... clean up test data from the database ... } public SqlConnection Db { get; private set; }
}
public class MyDatabaseTests : IClassFixtureDatabaseFixture
{ DatabaseFixture fixture; public MyDatabaseTests(DatabaseFixture fixture) { this.fixture fixture; } [Fact] public async Task GetTest() { // ... write tests, using fixture.Db to get access to the SQL Server ... // ... 在这里使用注入 的 DatabaseFixture }
}Singleton Sample这里以一个对 asp.net core API 的测试为例自定义 Fixture/// summary
/// Shared Context https://xunit.github.io/docs/shared-context.html
/// /summary
public class APITestFixture : IDisposable
{ private readonly IWebHost _server; public IServiceProvider Services { get; } public HttpClient Client { get; } public APITestFixture() { var baseUrl $http://localhost:{GetRandomPort()}; _server WebHost.CreateDefaultBuilder() .UseUrls(baseUrl) .UseStartupTestStartup() .Build(); _server.Start(); Services _server.Services; Client new HttpClient(new WeihanLi.Common.Http.NoProxyHttpClientHandler()) { BaseAddress new Uri(${baseUrl}) }; // Add Api-Version Header // Client.DefaultRequestHeaders.TryAddWithoutValidation(Api-Version, 1.2); Initialize(); Console.WriteLine(test begin); } /// summary /// TestDataInitialize /// /summary private void Initialize() { } public void Dispose() { using (var dbContext Services.GetRequiredServiceReservationDbContext()) { if (dbContext.Database.IsInMemory()) { dbContext.Database.EnsureDeleted(); } } Client.Dispose(); _server.Dispose(); Console.WriteLine(test end); } private static int GetRandomPort() { var random new Random(); var randomPort random.Next(10000, 65535); while (IPGlobalProperties.GetIPGlobalProperties().GetActiveTcpListeners().Any(p p.Port randomPort)) { randomPort random.Next(10000, 65535); } return randomPort; }
}
[CollectionDefinition(APITestCollection)]
public class APITestCollection : ICollectionFixtureAPITestFixture
{
}自定义Collection[CollectionDefinition(TestCollection)]
public class TestCollection : ICollectionFixtureTestStartupFixture
{
}自定义一个 TestBase[Collection(APITestCollection)]
public class ControllerTestBase
{ protected HttpClient Client { get; } protected IServiceProvider Services { get; } public ControllerTestBase(APITestFixture fixture) { Client fixture.Client; Services fixture.Services; }
}需要依赖注入的Test类写法public class NoticeControllerTest : ControllerTestBase
{ public NoticeControllerTest(APITestFixture fixture) : base(fixture) { } [Fact] public async Task GetNoticeList() { using (var response await Client.GetAsync(/api/notice)) { Assert.Equal(HttpStatusCode.OK, response.StatusCode); var responseString await response.Content.ReadAsStringAsync(); var result JsonConvert.DeserializeObjectPagedListModelNotice(responseString); Assert.NotNull(result); } } [Fact] public async Task GetNoticeDetails() { var path test-notice; using (var response await Client.GetAsync($/api/notice/{path})) { Assert.Equal(HttpStatusCode.OK, response.StatusCode); var responseString await response.Content.ReadAsStringAsync(); var result JsonConvert.DeserializeObjectNotice(responseString); Assert.NotNull(result); Assert.Equal(path, result.NoticeCustomPath); } } [Fact] public async Task GetNoticeDetails_NotFound() { using (var response await Client.GetAsync(/api/notice/test-notice1212)) { Assert.Equal(HttpStatusCode.NotFound, response.StatusCode); } }
}运行测试查看我们的 APITestFixture 是不是只实例化了一次查看输出日志可以看到我们输出的日志只有一次说明在整个测试过程中确实只实例化了一次只会启动一个 web server确实是单例的Memo微软推荐的是用 Microsoft.AspNetCore.Mvc.Testing 组件去测试 Controller但是个人感觉不如自己直接去写web 服务去测试如果没必要引入自己不熟悉的组件最好还是不要去引入新的东西否则可能就真的是踩坑不止了。Referencehttps://xunit.github.io/docs/shared-context.htmlhttps://github.com/WeihanLi/ActivityReservation/tree/dev/ActivityReservation.API.Test