成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

ASP.NET MVC單元測試:HttpContext類的Path屬性解惑

開發 后端
HttpContext類中有各種千奇百怪的Path屬性,很難掌握。本文談了一些對于HttpContext類Path屬性的一些理解,以便更好的進行單元測試。

有關HttpContext類的Path屬性問題描述

前一段時間有朋友在郵件中向我抱怨,說他們團隊在使用ASP.NET MVC開發時,在單元測試的時候總是遇到一些不那么方便的地方。例如,對于HttpContext中各種千奇百怪的Path總是無法掌控。例如某個功能會用到HttpContext的Path屬性,有的又要用到RawUrl——有的又會涉及到HostName。于是在單元測試的時候,就可能需要填充Mock對象的多種Path屬性,而這幾種Path屬性的值,在理論上還有關系。這其實還是小事,一個麻煩的事情在于,如果功能實現的方式變了,例如原本使用RawUrl屬性,而后來忽然覺得應該使用CurrentExecutionFilePath比較合適,于是單元測試就必須跟著改。如此反復,疲于奔命。

就我個人經驗看來,這種情況還是蠻常見的,因為某些時候兩種Path屬性的值差不多,看上去都可以正常使用,于是剛開始編寫的時候可能選擇了其中一個。但是后來發現,在另一些情況下兩種Path就有區別了,而且應該使用的是另一個屬性,于是不得不修改,進而單元測試失敗了。于是他問我,有沒有什么好方法來“完整而可靠地”設置那些繽繁復雜的Path屬性。我之前其實也是根據需求設置各種Path屬性,但是這的確不好,最重要的問題在于“單元測試”需要了解太多“被測試方法”的實現細節了,這種依賴非常的不可靠。雖然這也是Mock對象被人詬病的特點之一,但是如果我們能夠緩解這個缺陷自然再好不過了。

不過話說回來,在“應對”這個問題之前,您要先了解目前的功能是不是真要訪問HttpContext中的各種Path。ASP.NET MVC為了提高程序的可測試性作了很多努力,或者說,將“關注點”進行了很大程度的分離。在大部分情況下,我們都能夠不去觸及HttpContext,而且我們應該盡可能避免這種情況的發生。例如,對Controller做單元測試的時候直接傳遞參數,為Model Binder做單元測試的時候使用ValueProvider。想來想去,會直接使用到HttpContext的Path屬性的場景不多,可能自定義Route算是一個吧,因為它的功能就是解析URL。

HttpContext類的Path屬性原理

HttpContext的Path屬性都是通過HttpRequest對象獲得的。而事實上ASP.NET中的HttpRequest對象已經為我們提供一種直接通過URL構造的功能:

  1. var request = new HttpRequest(  
  2.     "",                                      /* filename */ 
  3.     "http://www.cnblogs.com/JeffreyZhao/",   /* url */ 
  4.     "hello=world");                          /* querystring */ 

估計ASP.NET開發團隊也知道URL是個難辦的問題,為我們預留了這樣一個構造函數。這時的request對象會預填了大多數Path相關的屬性:

  1. request  
  2. {System.Web.HttpRequest}  
  3.     AcceptTypes: null 
  4.     AnonymousID: null 
  5.     ApplicationPath: null 
  6.     AppRelativeCurrentExecutionFilePath: threw an exception of type 'System.NullReferenceException' 
  7.     Browser: null 
  8.     ClientCertificate: threw an exception of type 'System.NullReferenceException' 
  9.     ContentEncoding: threw an exception of type 'System.NullReferenceException' 
  10.     ContentLength: 0  
  11.     ContentType: "" 
  12.     Cookies: {System.Web.HttpCookieCollection}  
  13.     CurrentExecutionFilePath: "/JeffreyZhao/" 
  14.     FilePath: "/JeffreyZhao/" 
  15.     Files: {System.Web.HttpFileCollection}  
  16.     Filter: {System.Web.HttpInputStreamFilterSource}  
  17.     Form: {}  
  18.     Headers: {}  
  19.     HttpMethod: "GET" 
  20.     InputStream: {System.Web.HttpInputStream}  
  21.     IsAuthenticated: threw an exception of type 'System.NullReferenceException' 
  22.     IsLocal: false 
  23.     IsSecureConnection: false 
  24.     LogonUserIdentity: null 
  25.     Params: {hello=world}  
  26.     Path: "/JeffreyZhao/" 
  27.     PathInfo: "" 
  28.     PhysicalApplicationPath: threw an exception of type 'System.ArgumentNullException' 
  29.     PhysicalPath: "" 
  30.     QueryString: {hello=world}  
  31.     RawUrl: "/JeffreyZhao/?hello=world" 
  32.     RequestType: "GET" 
  33.     ServerVariables: {}  
  34.     TotalBytes: 0  
  35.     Url: {http://www.cnblogs.com/JeffreyZhao/}  
  36.     UrlReferrer: null 
  37.     UserAgent: null 
  38.     UserHostAddress: null 
  39.     UserHostName: null 
  40.     UserLanguages: null 

以上內容是從Visual Studio的Immediate Window中看到的,由此可以發現,其中大部分的Path屬性已經準備好了,但是AppRelativeCurrentExecutionFilePath屬性拋出異常(還有兩個與本地磁盤路徑有關的Path就忽略了),因為它需要特定的虛擬路徑環境才能計算出來。通過.NET Reflector觀察這個屬性的實現,會發現其中牽涉到的內容不是一點兩點,幾乎不可能通過設置外部環境的方式來使其通過。因此,我們最終還是要通過Mock框架來進行設置——反正我們也需要設置HttpRequest的其它屬性,不是嗎?

  1. var realRequest = new HttpRequest(  
  2.     "",                                      /* filename */ 
  3.     "http://www.cnblogs.com/JeffreyZhao/",   /* url */ 
  4.     "hello=world");                          /* querystring */ 
  5. var mockRequest = new Mock<HttpRequestWrapper>(realRequest) { CallBase = true };  
  6. mockRequest  
  7.     .Setup(r => r.AppRelativeCurrentExecutionFilePath)  
  8.     .Returns("~" + realRequest.CurrentExecutionFilePath);  

這里還是使用Moq框架,而Mock的對象則是HttpRequestWrapper類型,而不是我們常用的HttpRequestBase類型。HttpRequestWrapper的特點便是可以“塞入”一個真正的HttpRequest對象,然后把所有成員都委托給這個HttpRequest對象。我們在構建一個Mock<HttpRequestWrapper>對象之后,還需要把CallBase屬性設為true,這樣便可以讓Mock對象在默認情況下直接使用Wrapper的實現了。

有了Request,我們便可以構建一個HttpContext的Mock對象:

  1. var mockContext = new Mock<HttpContextBase>();  
  2. mockContext.Setup(c => c.Request).Returns(mockRequest.Object);  

但是,Moq框架有個限制,那就是如果您指定了這里的Request對象,再去通過HttpContext指定Request中的其他屬性,就會把原來的HttpRequest對象給覆蓋。也就是說,下面的代碼會讓我們對HttpRequest做的努力付之東流:

  1. mockContext.Setup(c => c.Request.Form).Returns(new NameValueCollection());  

這樣您會發現,mockContext.Object.Request下除了Form外的其他屬性都沒有值了(或拋出異常,視您Mock時的Behavior是Loose還是Strict而定)。因此,如果我們希望進一步修改HttpRequest中屬性的時候,只能直接使用那個Mock<HttpRequestWrapper>對象進行設置。我不清楚其他Mock框架的行為如何,如果您使用的也是Moq框架,可能就只得這么做了。

為了使用方便,我也在測試項目中準備了這樣一個輔助方法:

  1. public static class MockHelper  
  2. {  
  3.     public static Mock<HttpContextBase> MockRequest(string url, out Mock<HttpRequestWrapper> mockRequest)  
  4.     {  
  5.         int index = url.IndexOf('?');  
  6.         string path = index >= 0 ? url.Substring(0, index) : url;  
  7.         string queryString = index >= 0 ? url.Substring(index + 1) : "";  
  8.  
  9.         var realRequest = new HttpRequest("", path, queryString);  
  10.         mockRequest = new Mock<HttpRequestWrapper>(realRequest) { CallBase = true };  
  11.         mockRequest  
  12.             .Setup(r => r.AppRelativeCurrentExecutionFilePath)  
  13.             .Returns("~" + realRequest.CurrentExecutionFilePath);  
  14.  
  15.         var mockContext = new Mock<HttpContextBase>();  
  16.         mockContext.Setup(c => c.Request).Returns(mockRequest.Object);  
  17.         return mockContext;  
  18.     }  
  19. }  
  20.  

于是我們就可以更方便地進行相關的單元測試。例如,我們“象征性”地測試一下ASP.NET Routing中內置的Route類型:

  1. [Fact]  
  2. public void URL_Capturing_and_Generation()  
  3. {  
  4.     // prepare route  
  5.     Route route = new Route("{controller}/{action}/{id}"null);  
  6.  
  7.     // Mock request  
  8.     string url = "http://www.cnblogs.com/Home/Index/5";  
  9.     Mock<HttpRequestWrapper> mockRequest;  
  10.     var mockContext = MockHelper.MockRequest(url, out mockRequest);  
  11.     mockContext.Setup(c => c.Response.Charset).Returns("utf-8"); // if you need  
  12.  
  13.     // test data capturing  
  14.     RouteData routeData = route.GetRouteData(mockContext.Object);  
  15.     Assert.Equal("Home", routeData.GetRequiredString("controller"));  
  16.     Assert.Equal("Index", routeData.GetRequiredString("action"));  
  17.     Assert.Equal("5", routeData.GetRequiredString("id"));  
  18.  
  19.     // test url generation  
  20.     var hash = new { controller = "Account", action = "List", id = 1};  
  21.     var values = new RouteValueDictionary(hash);  
  22.     var requestContext = new RequestContext(mockContext.Object, routeData);  
  23.     var pathData = route.GetVirtualPath(requestContext, values);  
  24.     Assert.Equal("Account/List/1", pathData.VirtualPath);  
  25. }  
  26.  

具體內容就敘述到這里,目前Path相關的問題應該已經不會給您造成太大問題了。

以上就是對HttpContext類的Path屬性的問題解惑。本文來自老趙點滴:《在單元測試時指定HttpContext的各種Path》

【編輯推薦】

  1. 自定義的ControllerFactory:接口實現,支持Area
  2. ASP.NET Routing之“解析URL”功能詳解
  3. 為ASP.NET MVC應用添加自定義路由
  4. 學習ASP.NET MVC路由的使用方法
  5. 淺析ASP.NET中的URL Rewrite
責任編輯:yangsai 來源: 老趙點滴
相關推薦

2009-07-24 11:33:12

MVC單元測試ASP.NET

2009-07-23 16:29:06

ASP.NET單元測試

2009-06-01 09:13:52

ASP.NET MVCMVC應用ASP.NET MVC

2021-04-26 14:25:39

ASP.NET Cor單元測試

2021-05-11 15:50:28

ASP.NET單元測試

2009-07-31 12:43:59

ASP.NET MVC

2009-07-24 13:20:44

MVC框架ASP.NET

2009-07-20 15:44:32

ASP.NET MVC

2009-07-22 09:11:02

Action方法ASP.NET MVC

2009-07-23 17:07:58

2009-07-28 13:17:09

EnableViewSASP.NET

2009-07-22 13:24:24

ASP.NET MVC

2009-07-22 10:09:59

ASP.NET MVC

2009-07-23 14:31:20

ASP.NET MVC

2009-07-23 15:44:39

ASP.NET MVC

2009-07-20 10:53:59

ASP.NET MVC

2009-07-22 10:34:37

ActionInvokASP.NET MVC

2011-09-22 10:58:56

ASP.NET

2009-07-20 12:59:53

ASP.NET MVCASP.NET框架的功

2009-08-13 11:44:25

ASP.NET中的多種
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩一区二区在线视频 | 国产区精品| 成人在线观看中文字幕 | 91伊人 | 欧洲精品码一区二区三区免费看 | 91久久精品视频 | 亚洲精品一级 | 国际精品鲁一鲁一区二区小说 | 日韩免费一区二区 | 黄色免费在线观看网址 | jlzzjlzz欧美大全 | www.一区二区三区 | 99久久日韩精品免费热麻豆美女 | 精品一级 | 黄色av免费网站 | 国产在线网址 | 精品国产一区二区三区四区在线 | 精品91av| 久久99蜜桃综合影院免费观看 | 欧美日韩综合视频 | 超碰97av | 久草成人 | 日韩欧美久久 | 免费在线视频精品 | 久久久国产视频 | 精品久久国产 | 色伊人久久 | 国产精品一区二区三区99 | 中文字幕亚洲视频 | 成人在线免费观看视频 | 成人福利网 | 一级毛片网 | av毛片 | 国产精品福利视频 | 精品www| 激情91| 亚洲欧美日韩网站 | 99免费| 成人免费一区二区三区视频网站 | 免费看的黄网站 | 免费一级黄色 |