相關博文:《asp.net 5 使用 TestServer 進行單元測試》
在上一篇博文中,主要說的是,使用 TestServer 對 ASP.NET 5 WebApi 進行單元測試,依賴注入在 WebApi Startup.cs 中完成,所以 UnitTest 中只需要使用 TestServer 啟動 WebApi 站點就可以了,因為整個解決方案的對象都是用 ASP.NET 5 “自帶”的依賴注入進行管理,所以,在對 ASP.NET 5 類庫進行單元測試的時候,我都是手動進行 new 創建的對象,比如針對 application 的單元測試,貼一段 AdImageServiceTest 中的代碼:
namespace CNBlogs.Ad.Application.Tests{ public class AdTextServiceTest : BaseTest { PRivate IAdTextService _adTextService; public AdTextServiceTest(ITestOutputHelper output) : base(output) { Bootstrapper.Startup.ConfigureMapper(); IUnitOfWork unitOfWork = new UnitOfWork(dbContext); IMemcached memcached = new EnyimMemcached(null); _adTextService = new AdTextService(new AdTextRepository(dbContext), new AdTextCreativeRepository(dbContext), new UserService(memcached), unitOfWork, memcached); } [Fact] public async Task GetByAdTextsTest() { var adTexts = await _adTextService.GetAdTexts(); Assert.NotNull(adTexts); adTexts.ForEach(x => Console.WriteLine($"{x.Title}({x.Link})")); } }}
AdImageServiceTest 構造函數中的代碼,是不是看起來很別扭呢?我當時這樣寫,也是沒有辦法的,因為依賴注入的配置是寫在 Startup 中,比如下面代碼:
namespace CNBlogs.Ad.WebApi{ public class Startup { public Startup(IHostingEnvironment env, IApplicationEnvironment appEnv) { // Set up configuration sources. var builder = new ConfigurationBuilder() .AddJsonFile("appsettings.json") .AddEnvironmentVariables(); Configuration = builder.Build(); } public IConfigurationRoot Configuration { get; set; } // This method gets called by the runtime. Use this method to add services to the container. public void ConfigureServices(IServiceCollection services) { // Add framework services. services.AddMvc(); services.AddSingleton<IUnitOfWork, UnitOfWork>(); services.AddScoped<IDbContext, EFDbContext>(); services.AddSingleton<IAdTextRepository, AdTextRepository>(); services.AddSingleton<IAdTextService, AdTextService>(); } }}
這樣設計導致的結果是,針對類庫項目的單元測試,就沒辦法使用依賴注入獲取對象了,我后來想使用針對 WebApi 單元測試的方式,來對類庫進行單元測試,比如用 TestServer 來啟動,但類庫中沒有辦法獲取所注入的對象,構成函數注入會報錯,[FromServices]
屬性注入是 MVC Controller 中的東西,并不支持,所以針對類庫的單元測試,和 WebApi 的單元測試并不是一樣。
IServiceCollection 的程序包是 Microsoft.Extensions.DependencyInjection.Abstractions
,我原來以為它和 ASP.NET 5 Web 應用程序相關,其實它們也沒啥關系,你可以脫離 ASP.NET 5 Web 應用程序,獨立使用它,比如在類庫的單元測試中,但如果這樣設計使用,我們首先需要做一個工作,把 Startup.cs 中的 ConfigureServices 配置,獨立出來,比如放在 CNBlogs.Ad.Bootstrapper
中,這樣 Web 應用程序和單元測試項目,都可以使用它,減少代碼的重復,比如我們可以進行下面設計:
namespace CNBlogs.Ad.Bootstrapper{ public static class Startup { public static void Configure(this IServiceCollection services, string connectionString) { services.AddEntityFramework() .AddSqlServer() .AddDbContext<EFDbContext>(options => options.UseSqlServer(connectionString)); services.AddEnyimMemcached(); ConfigureMapper(); services.AddSingleton<IUnitOfWork, UnitOfWork>(); services.AddScoped<IDbContext, EFDbContext>(); services.AddSingleton<IAdTextRepository, AdTextRepository>(); services.AddSingleton<IAdTextService, AdTextService>(); } public static void ConfigureMapper() { Mapper.CreateMap<CNBlogs.Ad.Domain.Entities.AdText, AdTextDTO>(); } }}
ASP.NET 5 WebApi 項目中的 Startup.cs 配置會非常簡單,只需要下面代碼:
public void ConfigureServices(IServiceCollection services){ services.Configure(Configuration["data:ConnectionString"]);//add using CNBlogs.Ad.Bootstrapper;}
好了,做好上面工作后,單元測試中使用依賴注入就非常簡單了,為了減少重復代碼,我們可以先抽離出一個 BaseTest:
namespace CNBlogs.Ad.BaseTests{ public class BaseTest { protected readonly ITestOutputHelper output; protected IServiceProvider provider; public BaseTest(ITestOutputHelper output) { var connectionString = ""; var services = new ServiceCollection(); this.output = output; services.Configure(connectionString);////add using CNBlogs.Ad.Bootstrapper; provider = services.BuildServiceProvider(); } }}
可以看到,我們并沒有使用 TestServer,而是手動創建一個 ServiceCollection,它有點類似于我們之前寫依賴注入的 Unity Container,不過它們有很大不同,ServiceCollection 的功能更加強大,從 Bootstrapper.Startup 中可以看到,它可以注入 EF、MVC、Memcache 等等服務對象,BuildServiceProvider 的作用就是獲取注入的服務對象,我們下面會用到:
namespace CNBlogs.Ad.Repository.Tests{ public class AdTextServiceTest : BaseTest { private IAdTextService _adTextService; public AdTextServiceTest(ITestOutputHelper output) : base(output) { _adTextService = provider.GetService<IAdTextService>(); } [Fact] public async Task GetByAdTextsTest() { var adTexts = await _adTextService.GetAdTexts(); Assert.NotNull(adTexts); adTexts.ForEach(x => Console.WriteLine($"{x.Title}({x.Link})")); } }}
這段代碼和一開始的那段代碼,形成了鮮明對比,這就是代碼設計的魅力所在!?。?/p>
新聞熱點
疑難解答