【发布时间】:2022-03-28 13:01:54
【问题描述】:
是否有可能以某种方式拦截日志记录(SLF4J + logback)并通过 JUnit 测试用例获得InputStream(或其他可读的内容)...?
【问题讨论】:
是否有可能以某种方式拦截日志记录(SLF4J + logback)并通过 JUnit 测试用例获得InputStream(或其他可读的内容)...?
【问题讨论】:
Slf4j API 没有提供这种方式,但 Logback 提供了一个简单的解决方案。
您可以使用 ListAppender :一个白盒 logback appender,其中日志条目被添加到我们可以用来进行断言的 public List 字段中。
这是一个简单的例子。
Foo 类:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class Foo {
static final Logger LOGGER = LoggerFactory.getLogger(Foo .class);
public void doThat() {
logger.info("start");
//...
logger.info("finish");
}
}
FooTest 类:
import org.slf4j.LoggerFactory;
import ch.qos.logback.classic.Level;
import ch.qos.logback.classic.Logger;
import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.read.ListAppender;
public class FooTest {
@Test
void doThat() throws Exception {
// get Logback Logger
Logger fooLogger = (Logger) LoggerFactory.getLogger(Foo.class);
// create and start a ListAppender
ListAppender<ILoggingEvent> listAppender = new ListAppender<>();
listAppender.start();
// add the appender to the logger
fooLogger.addAppender(listAppender);
// call method under test
Foo foo = new Foo();
foo.doThat();
// JUnit assertions
List<ILoggingEvent> logsList = listAppender.list;
assertEquals("start", logsList.get(0)
.getMessage());
assertEquals(Level.INFO, logsList.get(0)
.getLevel());
assertEquals("finish", logsList.get(1)
.getMessage());
assertEquals(Level.INFO, logsList.get(1)
.getLevel());
}
}
您还可以将 Matcher/assertion 库用作 AssertJ 或 Hamcrest。
使用 AssertJ 会是:
import org.assertj.core.api.Assertions;
Assertions.assertThat(listAppender.list)
.extracting(ILoggingEvent::getFormattedMessage, ILoggingEvent::getLevel)
.containsExactly(Tuple.tuple("start", Level.INFO), Tuple.tuple("finish", Level.INFO));
【讨论】:
Logger fooLogger = (Logger) LoggerFactory.getLogger(Foo.class); 的 ClassCastException。我正在使用LoggerFactory 的org.slf4j.LoggerFactory 和Logger 的ch.qos.logback.classic.Logger
org.junit.platform.commons.logging.LoggerFactory,这就是发生在我身上的事情。
SLF4J,此解决方案最终会引发SLF4J: Class path contains multiple SLF4J bindings. 警告,因为您同时拥有 SLF4J 和 logback.classic
您可以创建自定义附加程序
public class TestAppender extends AppenderBase<LoggingEvent> {
static List<LoggingEvent> events = new ArrayList<>();
@Override
protected void append(LoggingEvent e) {
events.add(e);
}
}
并配置 logback-test.xml 以使用它。现在我们可以从我们的测试中检查日志事件:
@Test
public void test() {
...
Assert.assertEquals(1, TestAppender.events.size());
...
}
注意:如果您没有得到任何输出,请使用 ILoggingEvent - 请参阅评论部分了解原因。
【讨论】:
ILoggingEvent 而不是LoggingEvent。这对我有用。
events。
sample0.xml 中提到的[此处] (logback.qos.ch/manual/configuration.html)。不要忘记将附加程序更改为您的实现
您可以使用来自http://projects.lidalia.org.uk/slf4j-test/ 的 slf4j-test。 它用它自己的 slf4j api 实现替换了整个 logback slf4j 实现以进行测试,并提供一个 api 来断言日志事件。
示例:
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<classpathDependencyExcludes>
<classpathDependencyExcludes>ch.qos.logback:logback-classic</classpathDependencyExcludes>
</classpathDependencyExcludes>
</configuration>
</plugin>
</plugins>
</build>
public class Slf4jUser {
private static final Logger logger = LoggerFactory.getLogger(Slf4jUser.class);
public void aMethodThatLogs() {
logger.info("Hello World!");
}
}
public class Slf4jUserTest {
Slf4jUser slf4jUser = new Slf4jUser();
TestLogger logger = TestLoggerFactory.getTestLogger(Slf4jUser.class);
@Test
public void aMethodThatLogsLogsAsExpected() {
slf4jUser.aMethodThatLogs();
assertThat(logger.getLoggingEvents(), is(asList(info("Hello World!"))));
}
@After
public void clearLoggers() {
TestLoggerFactory.clear();
}
}
【讨论】:
slf4j-test包的完整示例可以在这里找到:github.com/jaegertracing/jaeger-client-java/pull/378/files
private ListAppender<ILoggingEvent> logWatcher;
@BeforeEach
void setup() {
this.logWatcher = new ListAppender<>();
this.logWatcher.start();
((Logger) LoggerFactory.getLogger(MyClass.class)).addAppender(this.logWatcher);
}
@Test
void myMethod_logs2Messages() {
...
int logSize = logWatcher.list.size();
assertThat(logWatcher.list.get(logSize - 2).getFormattedMessage()).contains("EXPECTED MSG 1");
assertThat(logWatcher.list.get(logSize - 1).getFormattedMessage()).contains("EXPECTED MSG 2");
}
归功于:@davidxxx 的回答。查看import ch.qos.logback...详情:https://stackoverflow.com/a/52229629/601844
【讨论】:
一个简单的解决方案是使用 Mockito 模拟 appender(例如)
@Slf4j
class MyClass {
public void doSomething() {
log.info("I'm on it!");
}
}
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.is;
import static org.mockito.Mockito.verify;
@RunWith(MockitoJUnitRunner.class)
public class MyClassTest {
@Mock private Appender<ILoggingEvent> mockAppender;
private MyClass sut = new MyClass();
@Before
public void setUp() {
Logger logger = (Logger) LoggerFactory.getLogger(MyClass.class.getName());
logger.addAppender(mockAppender);
}
@Test
public void shouldLogInCaseOfError() {
sut.doSomething();
verify(mockAppender).doAppend(ArgumentMatchers.argThat(argument -> {
assertThat(argument.getMessage(), containsString("I'm on it!"));
assertThat(argument.getLevel(), is(Level.INFO));
return true;
}));
}
}
注意:我使用的是断言而不是返回 false,因为它使代码和(可能的)错误更易于阅读,但如果您有多个验证,它将不起作用。在这种情况下,您需要返回 boolean 指示该值是否符合预期。
【讨论】:
虽然创建自定义 logback appender 是一个很好的解决方案,但这只是第一步,你最终将开发/重新发明 slf4j-test,如果你走得更远一点:spf4j-slf4j-test 或其他我不知道的框架还不知道。
您最终将需要担心您在内存中保留了多少事件,在记录错误(而不是断言)时使单元测试失败,在测试失败时提供调试日志等等......
免责声明:我是 spf4j-slf4j-test 的作者,我编写这个后端是为了能够更好地测试 spf4j,这是一个很好的地方来查看如何使用 spf4j-slf4j-test 的示例。我实现的主要优势之一是减少了构建输出(Travis 限制了它),同时在发生故障时仍然拥有我需要的所有细节。
【讨论】:
我会推荐一个简单、可重用的间谍实现,它可以作为 JUnit 规则包含在测试中:
public final class LogSpy extends ExternalResource {
private Logger logger;
private ListAppender<ILoggingEvent> appender;
@Override
protected void before() {
appender = new ListAppender<>();
logger = (Logger) LoggerFactory.getLogger(Logger.ROOT_LOGGER_NAME); // cast from facade (SLF4J) to implementation class (logback)
logger.addAppender(appender);
appender.start();
}
@Override
protected void after() {
logger.detachAppender(appender);
}
public List<ILoggingEvent> getEvents() {
if (appender == null) {
throw new UnexpectedTestError("LogSpy needs to be annotated with @Rule");
}
return appender.list;
}
}
在您的测试中,您将通过以下方式激活间谍:
@Rule
public LogSpy log = new LogSpy();
调用log.getEvents()(或其他自定义方法)来检查记录的事件。
【讨论】:
import ch.qos.logback.classic.Logger; 而不是import org.slf4j.LoggerFactory; 否则addAppender() 不可用。我花了一段时间才弄清楚这一点。
before() 和 after() 从未达到,因此从未创建/附加附加程序并且 UnexpectedTestError 触发。任何想法我做错了什么?规则是否需要放入某个包中?另外,请将导入部分添加到您的答案中,因为某些对象/接口的名称不明确。
我在测试日志行时遇到问题,例如:LOGGER.error(message, exception)。
http://projects.lidalia.org.uk/slf4j-test/ 中描述的解决方案也尝试对异常进行断言,并且重新创建堆栈跟踪并不容易(在我看来毫无价值)。
我是这样解决的:
import org.junit.Test;
import org.slf4j.Logger;
import uk.org.lidalia.slf4jext.LoggerFactory;
import uk.org.lidalia.slf4jtest.TestLogger;
import uk.org.lidalia.slf4jtest.TestLoggerFactory;
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.groups.Tuple.tuple;
import static uk.org.lidalia.slf4jext.Level.ERROR;
import static uk.org.lidalia.slf4jext.Level.INFO;
public class Slf4jLoggerTest {
private static final Logger LOGGER = LoggerFactory.getLogger(Slf4jLoggerTest.class);
private void methodUnderTestInSomeClassInProductionCode() {
LOGGER.info("info message");
LOGGER.error("error message");
LOGGER.error("error message with exception", new RuntimeException("this part is not tested"));
}
private static final TestLogger TEST_LOGGER = TestLoggerFactory.getTestLogger(Slf4jLoggerTest.class);
@Test
public void testForMethod() throws Exception {
// when
methodUnderTestInSomeClassInProductionCode();
// then
assertThat(TEST_LOGGER.getLoggingEvents()).extracting("level", "message").contains(
tuple(INFO, "info message"),
tuple(ERROR, "error message"),
tuple(ERROR, "error message with exception")
);
}
}
这还具有不依赖 Hamcrest 匹配器库的优势。
【讨论】:
这是使用 lambdas 的替代方法,它使日志捕获逻辑在测试中可重用(封装其实现)并且不需要@BeforeEach/@AfterEach(在某些建议的解决方案中,附加程序未分离,这可以导致内存泄漏)。
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class MyService {
private static final Logger LOG = LoggerFactory.getLogger(MyService.class);
public void doSomething(String someInput) {
...
LOG.info("processing request with input {}", someInput);
...
}
}
package mypackage.util
import ch.qos.logback.classic.Logger;
import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.read.ListAppender;
import org.slf4j.LoggerFactory;
import java.util.List;
public class LogInterceptor {
public static List<ILoggingEvent> interceptLogs(Class<?> klass, Runnable runnable) {
final Logger logger = (Logger) LoggerFactory.getLogger(klass);
final ListAppender<ILoggingEvent> listAppender = new ListAppender<>();
listAppender.start();
logger.addAppender(listAppender);
try {
runnable.run();
return listAppender.list;
} finally {
logger.detachAppender(listAppender);
}
}
}
import static mypackage.util.LogInterceptor.interceptLogs;
public class MyServiceTest {
private MyService myService;
...
@Test
void doSomethingLogsLineWithTheGivenInput() {
List<ILoggingEvent> logs = interceptLogs(
myService.getClass(),
() -> myService.doSomething("foo")
);
assertThat(logs).isNotEmpty();
ILoggingEvent logEntry = logs.get(0);
assertThat(logEntry.getFormattedMessage()).isEqualTo("Processing request with input foo");
assertThat(logEntry.getLevel()).isEqualTo(Level.INFO);
}
}
【讨论】: