我正在尝试使用 mockito 验证记录器消息。
但是,我无法运行我的 junit 类来覆盖所有代码行。
你知道原因吗?
我的代码:
public class App {
private static final Logger LOGGER = Logger.getLogger(App.class);
public List<String> addToListIfSizeIsUnder3(final List<String> list, final String value) {
if (list == null) {
LOGGER.error("A null list was passed in");
return null;
}
if (list.size() < 3) {
list.add(value);
} else {
LOGGER.debug("The list already has {} entries"+ list.size());
}
return list;
}
}
我的 JUnit 类
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import java.util.ArrayList;
import java.util.List;
import org.apache.log4j.Appender;
import org.apache.log4j.Level;
import org.apache.log4j.Logger;
import org.apache.log4j.spi.LoggingEvent;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.ArgumentCaptor;
import org.mockito.Captor;
import org.mockito.Mock;
import org.mockito.runners.MockitoJUnitRunner;
@RunWith(MockitoJUnitRunner.class)
public class AppTest {
private App uut;
@Mock
private Appender mockAppender;
@Captor
private ArgumentCaptor<LoggingEvent> captorLoggingEvent;
@Before
public void setup() {
uut = new App();
Logger root = Logger.getRootLogger();
root.addAppender(mockAppender);
root.setLevel(Level.INFO);
}
/**
* I want to test with over 3 elements.
*/
@Test
public void testWithOver3Element() {
List<String> myList = new ArrayList<String>();
myList.add("value 1");
myList.add("value 2");
myList.add("value 3");
myList.add("value 4");
List<String> outputList = uut.addToListIfSizeIsUnder3(myList, "some value");
Assert.assertEquals(4, outputList.size());
Assert.assertFalse(myList.contains("some value"));
try {
verify(mockAppender, times(1)).doAppend(captorLoggingEvent.capture());
} catch (AssertionError e) {
e.printStackTrace();
}
LoggingEvent loggingEvent = captorLoggingEvent.getAllValues().get(0);
Assert.assertEquals("The list already has {} entries", loggingEvent.getMessage());
Assert.assertEquals(Level.DEBUG, loggingEvent.getLevel());
}
}
错误:
想要但未调用:mockAppender.doAppend(); -> 在 AppTest.testWithOver3Element(AppTest.java:52) 实际上,与此模拟的交互为零。
在 AppTest.testWithOver3Element(AppTest.java:52) 在 sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 在 sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) 在 sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) 在 java。 lang.reflect.Method.invoke(未知来源)在 org.junit.runners.model.FrameworkMethod\(1.runReflectiveCall(FrameworkMethod.java:47)在 org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java: 12) 在 org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44) 在 org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) 在 org.junit.internal.runners .statements.RunBefores.evaluate(RunBefores.java:26) 在 org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271) 在 org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70) 在 org. junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50) 在 org.junit.runners.ParentRunner\)3.run(Paren tRunner.java:238) 在 org.junit.runners.ParentRunner\(1.schedule(ParentRunner.java:63) 在 org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236) 在 org.junit.runners.ParentRunner。 access\)000(ParentRunner.java:53) 在 org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229) 在 org.junit.runners.ParentRunner.run(ParentRunner.java:309) 在 org.mockito.internal .runners.JUnit45AndHigherRunnerImpl.run(JUnit45AndHigherRunnerImpl.java:37) 在 org.mockito.runners.MockitoJUnitRunner.run(MockitoJUnitRunner.java:62) 在 org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java) :86) 在 org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459) 在org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:678) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382) 在 org.eclipse。 jdt.inte rnal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)
原文由 user3205761 发布,翻译遵循 CC BY-SA 4.0 许可协议
您可以做几件事来改进您的代码:
切换到 slf4j。这将允许您对接口进行编码,并且不知道幕后的日志记录实现(在您的情况下是 apache log4j)。
切换到 slf4j 将允许您在传递到日志记录框架时不必连接字符串 - 例如:
这具有使字符串文字中的占位符实际起作用的额外好处。
您正在尝试通过日志记录框架间接断言和捕获对对象的调用。这将是脆弱且容易出错的,因为您永远不知道日志框架内部会进行哪些调用。只模拟你的直接依赖。
不要测试日志记录语句。这不是类的完全可见的行为,它使测试变得脆弱和复杂。另外,像您所说的那样使用 ArrayList(即语言的一部分)来处理日志记录语句可以让它们得到充分的锻炼,并且它们将信息输出到控制台,这可能有助于调试失败的测试。脆弱的一个例子是,如果您更改日志记录语句以添加更多信息,或者您可能向该方法添加另一个日志记录语句,则此测试可能会无缘无故地中断。至少不要断言调用的次数,因为这将非常脆弱。
综上所述,如果您必须测试与 Logging 框架的交互 - 这里是您的代码的修改版本,可以运行并提供相同的功能。这基本上是改进列表中的选项 #3 -
您还可以查看 PowerMockito 之类的包来模拟静态 - 但只有在绝对需要时才可以。
希望这可以帮助。