一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - ASP.NET教程 - Asp.Net Core利用xUnit進行主機級別的網絡集成測試詳解

Asp.Net Core利用xUnit進行主機級別的網絡集成測試詳解

2020-06-05 16:25Ron.liang ASP.NET教程

這篇文章主要給大家介紹了關于Asp.Net Core利用xUnit進行主機級別的網絡集成測試的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們來一起看看吧

前言

在開發 Asp.Net Core 應用程序的過程中,我們常常需要對業務代碼編寫單元測試,這種方法既快速又有效,利用單元測試做代碼覆蓋測試,也是非常必要的事情;但是,但我們需要對系統進行集成測試的時候,需要啟動服務主機,利用瀏覽器或者Postman 等網絡工具對接口進行集成測試,這就非常的不方便,同時浪費了大量的時間在重復啟動應用程序上;今天要介紹就是如何在不啟動應用程序的情況下,對 Asp.Net Core WebApi 項目進行網絡集成測試。

一、建立項目

1.1 首先我們建立兩個項目,Asp.Net Core WebApi 和 xUnit 單元測試項目,如下

Asp.Net Core利用xUnit進行主機級別的網絡集成測試詳解

1.2 上圖的單元測試項目 Ron.XUnitTest 必須應用待測試的 WebApi 項目 Ron.TestDemo

1.3 接下來打開 Ron.XUnitTest 項目文件 .csproj,添加包引用

?
1
2
Microsoft.AspNetCore.App
Microsoft.AspNetCore.TestHost

1.4 為什么要引用這兩個包呢,因為我剛才創建的 WebApi 項目是引用 Microsoft.AspNetCore.App 的,至于 Microsoft.AspNetCore.TestHost,它是今天的主角,為了使用測試主機,必須對其進行引用,下面會詳細說明

二、編寫業務

2.1 創建一個接口,代碼如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
[Route("api/[controller]")]
[ApiController]
public class ValuesController : ControllerBase
{
private IConfiguration configuration;
public ValuesController(IConfiguration configuration)
{
 this.configuration = configuration;
}
 
[HttpGet("{id}")]
public ActionResult<int> Get(int id)
{
 var result= id + this.configuration.GetValue<int>("max");
 
 return result;
}
}

2.1 接口代碼非常簡單,接受一個參數 id,然后和配置文件中獲取的值 max 相加,然后輸出結果給客戶端

三、編寫測試用例

3.1 為了能夠使用主機集成測試,我們需要使用類

?
1
Microsoft.AspNetCore.TestHost.TestServer

3.2 我們來看一下 TestServer 的源碼,代碼較長,你可以直接跳過此段,進入下一節 3.3

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
public class TestServer : IServer
{
private IWebHost _hostInstance;
private bool _disposed = false;
private IHttpApplication<Context> _application;
 
public TestServer(): this(new FeatureCollection())
{
}
 
public TestServer(IFeatureCollection featureCollection)
{
 Features = featureCollection ?? throw new ArgumentNullException(nameof(featureCollection));
}
 
public TestServer(IWebHostBuilder builder): this(builder, new FeatureCollection())
{
}
 
public TestServer(IWebHostBuilder builder, IFeatureCollection featureCollection): this(featureCollection)
{
 if (builder == null)
 {
 throw new ArgumentNullException(nameof(builder));
 }
 
 var host = builder.UseServer(this).Build();
 host.StartAsync().GetAwaiter().GetResult();
 _hostInstance = host;
}
 
public Uri BaseAddress { get; set; } = new Uri("http://localhost/");
 
public IWebHost Host
{
 get
 {
 return _hostInstance
  ?? throw new InvalidOperationException("The TestServer constructor was not called with a IWebHostBuilder so IWebHost is not available.");
 }
}
 
public IFeatureCollection Features { get; }
 
private IHttpApplication<Context> Application
{
 get => _application ?? throw new InvalidOperationException("The server has not been started or no web application was configured.");
}
 
public HttpMessageHandler CreateHandler()
{
 var pathBase = BaseAddress == null ? PathString.Empty : PathString.FromUriComponent(BaseAddress);
 return new ClientHandler(pathBase, Application);
}
 
public HttpClient CreateClient()
{
 return new HttpClient(CreateHandler()) { BaseAddress = BaseAddress };
}
 
public WebSocketClient CreateWebSocketClient()
{
 var pathBase = BaseAddress == null ? PathString.Empty : PathString.FromUriComponent(BaseAddress);
 return new WebSocketClient(pathBase, Application);
}
 
public RequestBuilder CreateRequest(string path)
{
 return new RequestBuilder(this, path);
}
 
public async Task<HttpContext> SendAsync(Action<HttpContext> configureContext, CancellationToken cancellationToken = default)
{
 if (configureContext == null)
 {
 throw new ArgumentNullException(nameof(configureContext));
 }
 
 var builder = new HttpContextBuilder(Application);
 builder.Configure(context =>
 {
 var request = context.Request;
 request.Scheme = BaseAddress.Scheme;
 request.Host = HostString.FromUriComponent(BaseAddress);
 if (BaseAddress.IsDefaultPort)
 {
  request.Host = new HostString(request.Host.Host);
 }
 var pathBase = PathString.FromUriComponent(BaseAddress);
 if (pathBase.HasValue && pathBase.Value.EndsWith("/"))
 {
  pathBase = new PathString(pathBase.Value.Substring(0, pathBase.Value.Length - 1));
 }
 request.PathBase = pathBase;
 });
 builder.Configure(configureContext);
 return await builder.SendAsync(cancellationToken).ConfigureAwait(false);
}
 
public void Dispose()
{
 if (!_disposed)
 {
 _disposed = true;
 _hostInstance.Dispose();
 }
}
 
Task IServer.StartAsync<TContext>(IHttpApplication<TContext> application, CancellationToken cancellationToken)
{
 _application = new ApplicationWrapper<Context>((IHttpApplication<Context>)application, () =>
 {
 if (_disposed)
 {
  throw new ObjectDisposedException(GetType().FullName);
 }
 });
 
 return Task.CompletedTask;
}
 
Task IServer.StopAsync(CancellationToken cancellationToken)
{
 return Task.CompletedTask;
}
 
private class ApplicationWrapper<TContext> : IHttpApplication<TContext>
{
 private readonly IHttpApplication<TContext> _application;
 private readonly Action _preProcessRequestAsync;
 
 public ApplicationWrapper(IHttpApplication<TContext> application, Action preProcessRequestAsync)
 {
 _application = application;
 _preProcessRequestAsync = preProcessRequestAsync;
 }
 
 public TContext CreateContext(IFeatureCollection contextFeatures)
 {
 return _application.CreateContext(contextFeatures);
 }
 
 public void DisposeContext(TContext context, Exception exception)
 {
 _application.DisposeContext(context, exception);
 }
 
 public Task ProcessRequestAsync(TContext context)
 {
 _preProcessRequestAsync();
 return _application.ProcessRequestAsync(context);
 }
}
}

3.3 TestServer 類代碼量比較大,不過不要緊,我們只需要關注它的構造方法就可以了

?
1
2
3
4
public TestServer(IWebHostBuilder builder)
 : this(builder, new FeatureCollection())
{
}

3.4 其構造方法接受一個 IWebHostBuilder 對象,只要我們傳入一個 WebHostBuilder 就可以創建一個測試主機了

3.5 創建測試主機和 HttpClient 客戶端,我們在測試類 ValuesUnitTest 編寫如下代碼

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class ValuesUnitTest
{
private TestServer testServer;
private HttpClient httpCLient;
 
public ValuesUnitTest()
{
 testServer = new TestServer(new WebHostBuilder().UseStartup<Ron.TestDemo.Startup>());
 httpCLient = testServer.CreateClient();
}
 
[Fact]
public async void GetTest()
{
 var data = await httpCLient.GetAsync("/api/values/100");
 var result = await data.Content.ReadAsStringAsync();
 
 Assert.Equal("300", result);
}
}

代碼解釋

這段代碼非常簡單,首先,我們聲明了一個 TestServer 和 HttpClient 對象,并在構造方法中初始化他們; TestServer 的初始化是由我們 new 了一個 Builder 對象,并指定其使用待測試項目 Ron.TestDemo 中的 Startup 類來啟動,這樣我們能可以直接使用待測試項目的路由和管道了,甚至我們無需指定測試站點,因為這些都會在 TestServer 自動配置一個 localhost 的主機地址

3.7 接下來就是創建了一個單元測試的方法,直接使用剛才初始化的 HttpClient 對象進行網絡請求,這個時候,我們只需要知道 Action 即可,同時傳遞參數 100,最后斷言服務器輸出值為:"300",回顧一下我們創建的待測試方法,其業務正是將客戶端傳入的 id 值和配置文件 max 值相加后輸出,而 max 值在這里被配置為 200

3.8 運行單元測試

Asp.Net Core利用xUnit進行主機級別的網絡集成測試詳解

3.9 測試通過,可以看到,測試達到了預期的結果,服務器正確返回了計算后的值

四、配置文件注意事項

4.1 在待測試項目中的配置文件 appsettings.json 并不會被測試主機所讀取,因為我們在上面創建測試主機的時候沒有調用方法

?
1
WebHost.CreateDefaultBuilder

4.2 我們只是創建了一個 WebHostBuilder 對象,非常輕量的主機配置,簡單來說就是無配置,如果對于 WebHost.CreateDefaultBuilder 不理解的同學,建議閱讀我的文章 asp.netcore 深入了解配置文件加載過程.

4.3 所以,為了能夠在單元測試中使用項目配置文件,我在 Ron.TestDemo 項目中的 Startup 類加入了下面的代碼

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class Startup
{
public Startup(IConfiguration configuration, IHostingEnvironment env)
{
 this.Configuration = new ConfigurationBuilder()
 .AddJsonFile("appsettings.json")
 .AddEnvironmentVariables()
 .SetBasePath(env.ContentRootPath)
 .Build();
}
 
public IConfiguration Configuration { get; }
 
// This method gets called by the runtime. Use this method to add services to the container.
public void ConfigureServices(IServiceCollection services)
{
 services.AddSingleton<IConfiguration>(this.Configuration);
 services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
}
}

4.4 其目的就是手動讀取配置文件,重新初始化 IConfiguration 對象,并將 this.Configuration 對象加入依賴注入容器中

結語

  • 本文從單元測試入手,針對常見的系統集成測試提供了另外一種便捷的測試方案,通過創建 TestServer 測試主機開始,利用主機創建 HttpCLient 對象進行網絡集成測試
  • 減少重復啟動程序和測試工具,提高了測試效率
  • 充分利用了 Visual Studio 的優勢,既可以做單元測試,還能利用這種測試方案進行快速代碼調試
  • 最后,還了解如何通過 TestServer 主機加載待測試項目的配置文件對象 IConfiguration

示例代碼下載

Ron.TestDemo.rar

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。

原文鏈接:https://www.cnblogs.com/viter/p/10091816.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 69罗莉视频在线观看 | 91视在线国内在线播放酒店 | 国产综合图区 | 暖暖的免费观看高清视频韩国 | 美女胸又大又黄又www小说 | 亚洲波霸| 楚乔传第二部免费观看全集完整版 | 被18号每天强行榨干acg | 国外成品精品1688 | 99视频在线看观免费 | 亚洲国产精品综合久久网络 | 亚洲 欧美 中文 日韩欧美 | 久热这里在线精品 | 美女叽叽| 出轨娇妻的呻吟1—9 | 国产亚洲小视频 | 亚洲无线一二三四区 | 亚洲XXX午休国产熟女屁 | 女人扒开下面让男人桶爽视频 | 亚洲一区二区福利视频 | 国产另类视频一区二区三区 | 无码国产成人午夜在线观看不卡 | 欧美色综合高清免费 | 很黄的网站在线观看 | 亚洲2017久无码 | 天堂素人在线 | 国产目拍亚洲精品一区二区三区 | 四虎综合九九色九九综合色 | 久久成人a毛片免费观看网站 | 国产福利一区二区在线精品 | 希望影院高清免费观看视频 | 99久女女精品视频在线观看 | 国产精品第3页 | 猛男壮男受bl爽哭了高h | 丰满大屁股美女一级毛片 | 日本一级不卡一二三区免费 | 青青青久在线视频免费观看 | 欧美视频精品一区二区三区 | 国产成人咱精品视频免费网站 | 国产成人亚洲精品一区二区在线看 | 色婷婷影院在线视频免费播放 |