org.elasticsearch.test.MockLog Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of framework Show documentation
Show all versions of framework Show documentation
Elasticsearch subproject :test:framework
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/
package org.elasticsearch.test;
import org.apache.logging.log4j.Level;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.core.LogEvent;
import org.apache.logging.log4j.core.appender.AbstractAppender;
import org.apache.logging.log4j.core.config.Property;
import org.elasticsearch.common.logging.Loggers;
import org.elasticsearch.common.regex.Regex;
import org.elasticsearch.core.Releasable;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.regex.Pattern;
import static org.hamcrest.CoreMatchers.equalTo;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.is;
/**
* Test appender that can be used to verify that certain events were logged correctly
*/
public class MockLog implements Releasable {
private static final Map> mockLogs = new ConcurrentHashMap<>();
private static final MockAppender appender = new MockAppender();
private final List loggers;
private final List expectations;
private volatile boolean isAlive = true;
@Override
public void close() {
isAlive = false;
for (String logger : loggers) {
mockLogs.compute(logger, (k, v) -> {
assert v != null;
v.remove(this);
return v.isEmpty() ? null : v;
});
}
// check that all expectations have been evaluated before this is released
for (WrappedLoggingExpectation expectation : expectations) {
assertThat(
"Method assertMatched() not called on LoggingExpectation instance before release: " + expectation,
expectation.assertMatchedCalled,
is(true)
);
}
}
private static class MockAppender extends AbstractAppender {
MockAppender() {
super("mock", null, null, false, Property.EMPTY_ARRAY);
}
@Override
public void append(LogEvent event) {
List appenders = mockLogs.get(event.getLoggerName());
if (appenders == null) {
// check if there is a root appender
appenders = mockLogs.getOrDefault("", List.of());
}
for (MockLog appender : appenders) {
if (appender.isAlive == false) {
continue;
}
for (LoggingExpectation expectation : appender.expectations) {
expectation.match(event);
}
}
}
}
private MockLog(List loggers) {
/*
* We use a copy-on-write array list since log messages could be appended while we are setting up expectations. When that occurs,
* we would run into a concurrent modification exception from the iteration over the expectations in #append, concurrent with a
* modification from #addExpectation.
*/
expectations = new CopyOnWriteArrayList<>();
this.loggers = loggers;
}
/**
* Initialize the mock log appender with the log4j system.
*/
public static void init() {
appender.start();
Loggers.addAppender(LogManager.getLogger(""), appender);
}
public void addExpectation(LoggingExpectation expectation) {
expectations.add(new WrappedLoggingExpectation(expectation));
}
public void assertAllExpectationsMatched() {
for (LoggingExpectation expectation : expectations) {
expectation.assertMatched();
}
}
public interface LoggingExpectation {
void match(LogEvent event);
void assertMatched();
}
public abstract static class AbstractEventExpectation implements LoggingExpectation {
protected final String name;
protected final String logger;
protected final Level level;
protected final String message;
volatile boolean saw;
public AbstractEventExpectation(String name, String logger, Level level, String message) {
this.name = name;
this.logger = logger;
this.level = level;
this.message = message;
this.saw = false;
}
@Override
public void match(LogEvent event) {
if (event.getLevel().equals(level) && event.getLoggerName().equals(logger) && innerMatch(event)) {
if (Regex.isSimpleMatchPattern(message)) {
if (Regex.simpleMatch(message, event.getMessage().getFormattedMessage())) {
saw = true;
}
} else {
if (event.getMessage().getFormattedMessage().contains(message)) {
saw = true;
}
}
}
}
public boolean innerMatch(final LogEvent event) {
return true;
}
}
public static class UnseenEventExpectation extends AbstractEventExpectation {
public UnseenEventExpectation(String name, String logger, Level level, String message) {
super(name, logger, level, message);
}
@Override
public void assertMatched() {
assertThat("expected not to see " + name + " but did", saw, equalTo(false));
}
}
public static class SeenEventExpectation extends AbstractEventExpectation {
public SeenEventExpectation(String name, String logger, Level level, String message) {
super(name, logger, level, message);
}
@Override
public void assertMatched() {
assertThat("expected to see " + name + " but did not", saw, equalTo(true));
}
}
public static class EventuallySeenEventExpectation extends SeenEventExpectation {
private volatile boolean expectSeen = false;
public EventuallySeenEventExpectation(String name, String logger, Level level, String message) {
super(name, logger, level, message);
}
public void setExpectSeen() {
expectSeen = true;
}
@Override
public void assertMatched() {
if (expectSeen) {
super.assertMatched();
} else {
assertThat("expected not to see " + name + " yet but did", saw, equalTo(false));
}
}
}
public static class ExceptionSeenEventExpectation extends SeenEventExpectation {
private final Class extends Exception> clazz;
private final String exceptionMessage;
public ExceptionSeenEventExpectation(
final String name,
final String logger,
final Level level,
final String message,
final Class extends Exception> clazz,
final String exceptionMessage
) {
super(name, logger, level, message);
this.clazz = clazz;
this.exceptionMessage = exceptionMessage;
}
@Override
public boolean innerMatch(final LogEvent event) {
return event.getThrown() != null
&& event.getThrown().getClass() == clazz
&& event.getThrown().getMessage().equals(exceptionMessage);
}
}
public static class PatternSeenEventExpectation implements LoggingExpectation {
protected final String name;
protected final String logger;
protected final Level level;
protected final Pattern pattern;
volatile boolean saw;
public PatternSeenEventExpectation(String name, String logger, Level level, String pattern) {
this.name = name;
this.logger = logger;
this.level = level;
this.pattern = Pattern.compile(pattern);
}
@Override
public void match(LogEvent event) {
if (event.getLevel().equals(level) && event.getLoggerName().equals(logger)) {
if (pattern.matcher(event.getMessage().getFormattedMessage()).matches()) {
saw = true;
}
}
}
@Override
public void assertMatched() {
assertThat(name, saw, equalTo(true));
}
}
/**
* A wrapper around {@link LoggingExpectation} to detect if the assertMatched method has been called
*/
private static class WrappedLoggingExpectation implements LoggingExpectation {
private volatile boolean assertMatchedCalled = false;
private final LoggingExpectation delegate;
private WrappedLoggingExpectation(LoggingExpectation delegate) {
this.delegate = Objects.requireNonNull(delegate);
}
@Override
public void match(LogEvent event) {
delegate.match(event);
}
@Override
public void assertMatched() {
try {
delegate.assertMatched();
} finally {
assertMatchedCalled = true;
}
}
@Override
public String toString() {
return delegate.toString();
}
}
/**
* Adds the list of class loggers to this {@link MockLog}.
*
* Stops and runs some checks on the {@link MockLog} once the returned object is released.
*/
public static MockLog capture(Class>... classes) {
return create(Arrays.stream(classes).map(Class::getCanonicalName).toList());
}
/**
* Same as above except takes string class names of each logger.
*/
public static MockLog capture(String... names) {
return create(Arrays.asList(names));
}
private static MockLog create(List loggers) {
MockLog appender = new MockLog(loggers);
addToMockLogs(appender, loggers);
return appender;
}
private static void addToMockLogs(MockLog mockLog, List loggers) {
for (String logger : loggers) {
mockLogs.compute(logger, (k, v) -> {
if (v == null) {
v = new CopyOnWriteArrayList<>();
}
v.add(mockLog);
return v;
});
}
}
/**
* Executes an action and verifies expectations against the provided logger
*/
public static void assertThatLogger(Runnable action, Class> loggerOwner, MockLog.LoggingExpectation... expectations) {
try (var mockLog = MockLog.capture(loggerOwner)) {
for (var expectation : expectations) {
mockLog.addExpectation(expectation);
}
action.run();
mockLog.assertAllExpectationsMatched();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy