StrutsTestCase是用于測試Struts動作的強大易用的測試框架。結合傳統的JUnit測試,Struts及StrutsTestCase將為您提供高覆蓋率的測試,從而提高產品可靠性。
StrutsTestCase是基于JUnit的Struts動作測試框架。Struts將為測試應用程序的Struts動作類提供簡便有效的方法。
典型的J2EE應用程序是分層構建的,其結構如圖1所示:
◆DAO層封裝數據庫訪問。其中包括Hibernate映射、Object類、Hibernate查詢、實體EJB或其他實體-關系持久化技術。
◆業務層包含更多高級業務服務。理想狀態下,業務層相對獨立于數據庫實現。這一層常常用到EJB事務。
◆表示層向用戶展示應用程序數據,并解釋用戶請求。在Struts應用程序中,該層通常使用JSP/JSTL頁面顯示數據,并借助于Struts動作解釋用戶的查詢請求。
◆客戶端層主要是用戶機器上運行的web瀏覽器。客戶端邏輯(比如JavaScript)有時候置于該層,盡管很難對其進行有效的測試。
視具體架構,DAO及業務層可使用JUnit經典測試法或者各種JUnit擴展工具進行測試。DbUnit是進行數據庫單元測試的好選擇。
另一方面,通常很難對Struts動作進行測試。即使業務邏輯完全限定于業務層,Struts動作通常還是會包含重要的數據驗證、數據轉換和數據流控制代碼。如果不對Struts動作進行測試,那么將在代碼覆蓋率方面留下很大空白。而StrutsTestCase將填補這些空白。
對動作層進行單元測試還會帶來其他的一些好處:
◆視圖層及控制層設計起來更容易,通常也更為簡潔明了。
◆更易于重構動作類。
◆有助于避免冗余和無用的動作類。
◆測試用例有助于編寫動作層文檔,這些文檔在編寫JSP頁面時可以起到幫助作用。
這些是測試驅動開發的一些常見優點,這些優點適用于Struts動作層,也適用于其他的一些地方。
StrutsTestCase簡介
StrutsTestCase提供了在JUnit框架內測試Struts動作的靈活便利的方法。可以通過設置請求參數并檢查調用動作后生成的Request或Session狀態的方式,來對Struts動作進行白盒測試。
StrutsTestCase支持使用框架來模擬web服務器容器的模擬測試方法,或者在服務器容器(如Tomcat)內使用Cactus框架進行測試的容器內測試方法。一般來說,我更喜歡模擬測試方法,因為這種方法更為輕量級,運行更快,從而可以實現更緊湊的開發周期。
所有StrutsTestCase單元測試類均由模擬測試的MockStrutsTestCase或者容器內測試的CactusStrutsTestCase派生。由于模擬測試方法設置更簡單,運行更快,所以在這里我們將主要關注模擬測試方法。
StrutsTestCase實踐
為了使用StrutsTestCase測試該動作,我們將創建一個擴展MockStrutsTestCase類的新類。該類提供一些方法,用于構建模擬HTTP請求,調用相應的Struts動作,并在動作完成后驗證應用程序的狀態。
設想一個用于安排住宿的在線數據庫,它帶有一個復合條件搜索函數。搜索函數通過/search.do動作來實現。該動作將根據特定條件執行復合條件搜索,并將搜索結果列表置于名為results的屬性中,該屬性的作用域為請求范圍內。比如,下面的URL將會顯示法國的所有住宿地列表。
/search.do?country=FR
現在假設我們使用測試驅動方法來實現該方法。編寫動作類,更新Struts配置文件。編寫測試用例,測試該動作類(空)。采用嚴格的測試驅動開發方法,首先編寫測試用例,再實現符合測試用例的代碼。在實際情況中,具體順序視測試代碼不同而有所不同。
初始的測試用例看起來應該如下:
public void testSearchByCountry() {
setRequestPathInfo("/search.do");
addRequestParameter("country", "FR");
actionPerform();
}
在這里,我們設置調用路徑(setRequestPathInfo()),并添加請求參數(addRequestParameter())。
接下來,使用調用動作類。這樣將會驗證Struts配置并調用相應的動作類,但是并不會對動作的實際功能進行測試。為了測試動作的實際功能,我們需要驗證動作結果。
public void testSearchByCountry() {
setRequestPathInfo("/search.do");
addRequestParameter("country", "FR");
actionPerform();
verifyNoActionErrors();
verifyForward("suclearcase/" target="_blank" >ccess");
assertNotNull(request.getAttribute("results"));
}