junit不能测struts的action,httpunit也只能测servlet。用struts作项目的时候无法方便的对action层做单元测试一直是我的心头大恨。现在好了,我们有了StrutsTestCase。按照网上的介绍,StrutsTestCase用起来应该是非常简单的,只要下了jar包回来引用到工程里面就可以了。实际上可能也差不多--如果你运气不象我这么臭的话。
在sourceforge上随便挑其中一个镜象的下载地址:
http://aleron.dl.sourceforge.net/sourceforge/strutstestcase/strutstest213-1.2_2.4.zip
flashget回来,放到jbuilder的userhome里面,找一个struts action创建test case,创建的时候吧test case的基类改为 MockStrutsTestCase,测试的方法一个都不用选(因为我们是要针对action的具体每一个逻辑分支测试而不是具体的某一个方法)。创建成功后添加一个测试:
public void testSuccessfulRefresh()
{
setRequestPathInfo("/RefreshSystemData");
actionPerform();
verifyForward("success");
}
嘿嘿,我精心挑了一个没有参数的action来实验。
一切看起来很顺利。run test,噩梦开始了:
java.lang.NullPointerException
at servletunit.struts.MockStrutsTestCase.getActionServlet(MockStrutsTestCase.java:331)
at servletunit.struts.MockStrutsTestCase.tearDown(MockStrutsTestCase.java:130)
at hospital.tongren.oa.system.action.TestRefreshSystemDataAction.tearDown(TestRefreshSystemDataAction.java:34)
...(Click for full stack trace)...
还好我没有开音箱,不然又是一大炮轰出来。
看来要调试了,先去同一个地方下了StrutsTestCase原码回来
http://aleron.dl.sourceforge.net/sourceforge/strutstestcase/strutstest-213-src.zip
加进userhome里面的source。debug进去,跟到org.apache.struts.action.ActionServlet里面,出错的地方是:
InputStream input =
getServletContext().getResourceAsStream("/WEB-INF/web.xml");
try {
digester.parse(input);
} catch (IOException e) {
....
input 为空指针。不知道为什么ServletContextSimulator在模拟ServletContext的时候没能够正确的找到webmodule的位置。上网搜了好一会儿文档,在 http://strutstestcase.sourceforge.net/api/servletunit/struts/MockStrutsTestCase.html 中发现了这样一段:
NOTE: By default, the Struts ActionServlet will look for the file WEB-INF/struts-config.xml, so you must place the directory that contains WEB-INF in your CLASSPATH. ...
先照它说的试试把webmodule路径放进classpath中,没有用。
往下看,发现了这个好东东:setContextDirectory。在startup中加一句:
this.setContextDirectory(new File("E:\\projectPath\\webModulePath\\"));
终于把那个空指针给过了。但是报一个新的异常:
junit.framework.AssertionFailedError: received error 400 : Invalid path /RefreshSystemData was requested
at servletunit.HttpServletResponseSimulator.sendError(HttpServletResponseSimulator.java:463)
at org.apache.struts.action.RequestProcessor.processMapping(RequestProcessor.java:684)
at org.apache.struts.action.RequestProcessor.process(RequestProcessor.java:242)
at org.apache.struts.action.ActionServlet.process(ActionServlet.java:1482)
at org.apache.struts.action.ActionServlet.doPost(ActionServlet.java:525)
at servletunit.struts.MockStrutsTestCase.actionPerform(MockStrutsTestCase.java:394)
at hospital.tongren.oa.system.action.TestRefreshSystemDataAction.testSuccessfulRefresh(TestRefreshSystemDataAction.java:51)
...(Click for full stack trace)...
找不到strutsconfig里面配置的path。strutsconfig是在web.xml里面配置的,应该还是web.xml没找到造成的,那么就指定strutsconfig文件的位置咯:
setConfigFile("E:\\......\\struts-config.xml");
终于可以运行起来了。
随后发现,如果 setServletConfigFile("E:\\....\\WEB-INF\\web.xml");的话MockStrutsTestCase也能够根据web.xml中的配置找到strutsconfig文件。
最后把上面用到的绝对地址E:\\...全部改为相对地址:
setContextDirectory(new File("modulePath\\"));
setServletConfigFile("modulePath\\WEB-INF\\web.xml");
// this.setConfigFile("modulePath\\WEB-INF\\config\\system\\struts-config.xml");
血吐完了,继续郁闷,为什么别人都不用配置的这么麻烦呢?到底我做错了什么,还是jbuilder的错?
CactusStrutsTestCase也没配起,好像要加个什么包吧,再看看先。