Mike

导航

ASP.NET MVC Tip #25 – 单元测试 View 不使用 Web Server

ASP.NET MVC Tip #25 – 单元测试 View 不使用 Web Server
ASP.NET MVC Tip #25 – Unit Test Your Views without a Web Server

美语原文:http://weblogs.asp.net/stephenwalther/archive/2008/07/26/asp-net-mvc-tip-25-unit-test-your-views-without-a-web-server.aspx
国语翻译:http://www.cnblogs.com/mike108mvp

译者注:在下水平有限,翻译中若有错误或不妥之处,欢迎大家批评指正。谢谢。

译者注:ASP.NET MVC QQ交流群 1215279 欢迎对 ASP.NET MVC 感兴趣的朋友加入

在这篇帖子中,我将演示如何单元测试ASP.NET MVC views 而无需运行一个Web server。我将展示如何通过创建一个自定义的MVC View Engine和一个伪造的Controller Context来单元测试view。

你的web application可以测试的部分越多,你就对应用程序的修改不会带来新的bug越有信心。ASP.NET MVC让你很容易测试models 和controllers。在这篇帖子中,我将解释如何单元测试views。

创建一个自定义View Engine

让我们从创建一个自定义的View Engine开始。Listing 1中包含了一个叫做SimpleViewEngine的非常简单的View Engine。

Listing 1 – SimpleViewEngine.cs (C#)

namespace Tip25
{
    
public class SimpleViewEngine : IViewEngine
    
{
        
private string _viewsFolder = null;

        
public SimpleViewEngine()
        
{
            
if (HttpContext.Current != null)
            
{
                var root 
= HttpContext.Current.Request.PhysicalApplicationPath;
                _viewsFolder 
= Path.Combine(root, "Views");

            }

        }


        
public SimpleViewEngine(string viewsFolderPhysicalPath)
        
{
            _viewsFolder 
= viewsFolderPhysicalPath;
        }


        
public void RenderView(ViewContext viewContext)
        
{
            
if (_viewsFolder == null)
                
throw new NullReferenceException("You must supply a viewsFolder path");
            
string fullPath = Path.Combine(_viewsFolder, viewContext.ViewName) + ".htm";
            
if (!File.Exists(fullPath))
                
throw new HttpException(404"Page Not Found");

            
// Load file
            string rawContents = File.ReadAllText(fullPath);

            
// Perform replacements
            string parsedContents = Parse(rawContents, viewContext.ViewData);
            
            
// Write results to HttpContext
            viewContext.HttpContext.Response.Write(parsedContents);
        }


        
public string Parse(string contents, ViewDataDictionary viewData)
        
{
            
return Regex.Replace(contents, @"\{(.+)\}", m => GetMatch(m, viewData));
        }


        
protected virtual string GetMatch(Match m, ViewDataDictionary viewData)
        
{
            
if (m.Success)
            
{
                
string key = m.Result("$1");
                
if (viewData.ContainsKey(key))
                    
return viewData[key].ToString();
            }

            
return String.Empty;
        }


    }

}

注意SimpleViewEngine 实现了IViewEngine 接口。该接口有一个RenderView()方法必须实现。

在Listing 1中,RenderView()方法从硬盘中加载一个文件,并用ViewData中的项目来替换文件中的标签(tokens)。 在Listing 2中,包含了一个使用SimpleViewEngine的 controller 。当你调用HomeController.Index() action时,它返回一个叫做Index的view。

Listing 2 – HomeController.cs (C#)

namespace Tip25.Controllers
{
    [HandleError]
    
public class HomeController : Controller
    
{
        
public HomeController()
        
{
            
this.ViewEngine = new SimpleViewEngine();
        }



        
public ActionResult Index()
        
{
            ViewData[
"Message"= "Welcome to ASP.NET MVC!";
            ViewData[
"Message2"= "Using a custom View Engine";

            
return View("Index");
        }


    }

}

Index view在Listing 3中。注意文件名是Index.htm。SimpleViewEngine 返回了一个.htm 文件。

Listing 3 – Index.htm

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" >
<head>
    
<title>Tip 25</title>
</head>
<body>
    
    Here 
is the first message:
    
{message}

    
<br />
    Here 
is the second message:
    
<b>{message2}</b>

</body>
</html>

Index view包含也以前后大括号为标记的标签。SimpleViewEngine.RenderView()方法用View Data中的项目来替换每个同名标签。当Index view被SimpleViewEngine呈现时,你得到了Figure 1中的页面。

Figure 1 - Index view呈现的页面

创建一个伪造的Controller Context

SimpleViewEngine.RenderView()方法并不返回一个值。RenderView()方法将值直接写入HttpContext.Response对象中。因此,为了单元测试views,我们必须能够伪造HttpContext 对象,以便我们能够暗中监视添加到该对象中的值。

在我以前的两篇帖子中,我演示了如何伪造ControllerContext 和HttpContext 对象:

http://weblogs.asp.net/stephenwalther/archive/2008/06/30/asp-net-mvc-tip-12-faking-the-controller-context.aspx

http://weblogs.asp.net/stephenwalther/archive/2008/07/02/asp-net-mvc-tip-13-unit-test-your-custom-routes.aspx

在以前的这些帖子中,我演示了伪造ControllerContext 和 HttpContext对象是很有用的,当你需要单元测试Session State, Cookies, Form fields, 和 Route Tables时。

本文结尾处下载的代码中包含一个MvcFakes项目。我已经添加了一个伪造的HttpResponse 对象到一批伪造对象中,你可以在单元测试中使用这些伪造对象。

为View创建一个单元测试

既然我们已经创建了一个自定义的View Engine和一批伪造对象,那么我们就可以单元测试view了。Listing 4中的测试类测试了HomeController.Index() action返回的Index view。

Listing 4 – HomeControllerTest.cs (C#)


namespace Tip25Tests.Controllers
{
    
    [TestClass]
    
public class HomeControllerTest
    
{
        
private const string viewsPath =
            
@"C:\Users\swalther\Documents\Common Content\Blog\Tip25 Custom View Engine\CS\Tip25\Tip25\Views";

        [TestMethod]
        
public void Index()
        
{
            
// Setup controller
            HomeController controller = new HomeController();
            controller.ViewEngine 
= new SimpleViewEngine(viewsPath);

            
// Setup fake controller context
            var routeData = new RouteData();
            routeData.Values.Add(
"controller""home");
            var fakeContext 
= new FakeControllerContext(controller, routeData);

            
// Execute
            ViewResult result = controller.Index() as ViewResult;
            result.ExecuteResult(fakeContext);
            
string page = fakeContext.HttpContext.Response.ToString();

            
// Verify
            StringAssert.Contains(page, "<title>Tip 25</title>");
            StringAssert.Contains(page, 
"Welcome to ASP.NET MVC!""Missing Message");
            StringAssert.Contains(page, 
"<b>Using a custom View Engine</b>""Missing Message2 with bold");
        }


    }

}

Listing 4中的测试代码包含4个部分。第一部分通过将我们自定义的SimpleViewEngine 和HomeController 类联系起来,进行准备HomeController。注意你必须给SimpleViewEngine的构造函数提供一个硬编码的 view目录的路径。(下载本文的代码后,你需要修改这个路径)

第二部分准备伪造的ControllerContext 对象。注意你必须传递一个controller name 给FakeHttpContext 类的构造函数。

接着,HomeController.Index() action方法被调用。这个action 返回一个ViewResult。接着,该ViewResult 在伪造的HttpContext中被执行。最后,HttpResponse.ToString() 方法被调用来获得SimpleViewEngine写入HttpResponse 对象的值。

最后一部分证实view 呈现的页面包含三个子字符串。第一个是HTML 页面的title 被证实。第二,证实存在来自ViewData 的两个messages。注意该测试要证实第2个message 是否是粗体显示。

总结

在这篇帖子中,我展示了如何扩展你的单元测试来覆盖views。通过利用自定义的View Engine,你可以为models, controllers, 和 views创建单元测试。

下载代码:http://weblogs.asp.net/blogs/stephenwalther/Downloads/Tip25/Tip25.zip

posted on 2008-07-27 22:12  mike108mvp  阅读(2101)  评论(2编辑  收藏  举报