我必须在JEE应用程序中为少数servlet编写测试用例,一个简单的调用是使用JNDI使用多个bean和DataSource,问题是每次我模拟一个servelt,我在每个servelt调用的地方都会出错从servlet容器获取输入的任何东西。例如。
//servelts calls getDataSource()
public static synchronized DataSource getDataSource() throws NamingException {
return getDataSource(XYZConfig.DATA_SOURCE);
}
public static synchronized DataSource getDataSource(String dataSourceName) throws NamingException {
DataSource dataSource=null;
InitialContext initCntx=null;
try{
//TODO: remove
//hMap.remove(dataSourceName);
if(hMap.get(dataSourceName) != null){
dataSource=(DataSource) hMap.get(dataSourceName);
}else{
try{
initCntx=new InitialContext();
dataSource=(DataSource) initCntx.lookup(dataSourceName);
if(dataSource!=null){
hMap.put(dataSourceName,dataSource);
}
}catch(Exception ex){
dataSourceName = XYZConfig.DATA_SOURCE;
if(hMap.get(dataSourceName) != null){
dataSource=(DataSource) hMap.get(dataSourceName);
}else{
initCntx = new InitialContext();
dataSource=(DataSource) initCntx.lookup(FoursoftConfig.DATA_SOURCE);
if(dataSource!=null){
hMap.put(dataSourceName,dataSource);
}
}
}
}
}finally{
}
return dataSource;
}
现在我模拟了servlet,当从外部servelt容器调用时,我的servlet如何工作,NO JNDI被初始化,即使我这样做,我需要在源代码上稍微更改代码,我不能去做。并且它不仅仅是JNDI,同样适用于更少的bean。 非常感谢所有帮助。感谢。
我正在嘲笑servlet
HttpServletRequest request = mock(HttpServletRequest.class);
HttpServletResponse response = mock(HttpServletResponse.class);
when(request.getParameter("username")).thenReturn("user1");
when(request.getParameter("password")).thenReturn("pass1");
new UnitServelt().doPost(request,response);
例外情况如下:
映射controllerMap =(Map)getServletContext()。getAttribute(" controllerMap");
java.lang.IllegalStateException: ServletConfig has not been initialized
at javax.servlet.GenericServlet.getServletContext(GenericServlet.java:185)
at com.foursoft.etrans.common.servlets.ETransServlet.doPost(ETransServlet.java:56)
at com.foursoft.etrans.unitcase.MockRequest.testDoPostHttpServletRequestHttpServletResponse(MockRequest.java:57)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:47)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.junit.internal.runners.statements.RunAfters.evaluate(RunAfters.java:27)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:238)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.junit.internal.runners.statements.RunAfters.evaluate(RunAfters.java:27)
at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
如果我硬编码controllerMap,那么我在后续行
上得到例外 initCntx=new InitialContext();
dataSource=(DataSource) initCntx.lookup(dataSourceName);
为NoInitialContextException。
我确切地知道发生了什么,我的测试用例是从容器外部运行的,显然它不能有一个servletContext和任何来自容器的东西,我的问题是,我能做什么,整个应用程序正在使用HttpServletRequest无处不在,无处不在使用EJB和JNDI,我如何在容器外运行它时提供它。
答案 0 :(得分:1)
对此的正确答案取决于您是否尝试对整个servlet,其所有交互以及容器本身进行集成测试;或者你是否试图仅对servlet类进行单元测试。
要集成测试整个servlet,你可以启动在开发环境中运行的容器,并进行一系列实际发出HTTP请求的测试,并检查
为此使用JUnit是有道理的,但可能不需要Mockito。
另一方面,如果你所做的只是对servlet类进行单元测试,你应该模拟ServletConfig以及servlet类实际使用的容器API中的任何其他类。