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