All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.opensearch.test.MockLogAppender Maven / Gradle / Ivy

There is a newer version: 2.18.0
Show newest version
/*
 * SPDX-License-Identifier: Apache-2.0
 *
 * The OpenSearch Contributors require contributions made to
 * this file be licensed under the Apache-2.0 license or a
 * compatible open source license.
 */

/*
 * Licensed to Elasticsearch under one or more contributor
 * license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright
 * ownership. Elasticsearch licenses this file to you under
 * the Apache License, Version 2.0 (the "License"); you may
 * not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
/*
 * Modifications Copyright OpenSearch Contributors. See
 * GitHub history for details.
 */

package org.opensearch.test;

import org.apache.logging.log4j.Level;
import org.apache.logging.log4j.Logger;
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.apache.logging.log4j.core.filter.RegexFilter;
import org.opensearch.common.logging.Loggers;
import org.opensearch.common.regex.Regex;
import org.opensearch.test.junit.annotations.TestLogging;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.regex.Pattern;

import static org.hamcrest.CoreMatchers.equalTo;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.Assert.fail;

/**
 * Test appender that can be used to verify that certain events were logged correctly
 */
public class MockLogAppender extends AbstractAppender implements AutoCloseable {

    private static final String COMMON_PREFIX = System.getProperty("opensearch.logger.prefix", "org.opensearch.");

    private final List expectations;
    private final List loggers;

    /**
     * Creates an instance and adds it as an appender to the given Loggers. Upon
     * closure, this instance will then remove itself from the Loggers it was added
     * to. It is strongly recommended to use this class in a try-with-resources block
     * to guarantee that it is properly removed from all Loggers. Since the logging
     * state is static and therefore global within a JVM, it can cause unrelated
     * tests to fail if, for example, they trigger a logging statement that tried to
     * write to a closed MockLogAppender instance.
     */
    public static MockLogAppender createForLoggers(Logger... loggers) throws IllegalAccessException {
        final String callingClass = Thread.currentThread().getStackTrace()[2].getClassName();
        return createForLoggersInternal(callingClass, ".*(\n.*)*", loggers);
    }

    public static MockLogAppender createForLoggers(String filter, Logger... loggers) throws IllegalAccessException {
        final String callingClass = Thread.currentThread().getStackTrace()[2].getClassName();
        return createForLoggersInternal(callingClass, filter, loggers);
    }

    private static MockLogAppender createForLoggersInternal(String callingClass, String filter, Logger... loggers)
        throws IllegalAccessException {
        final MockLogAppender appender = new MockLogAppender(
            callingClass + "-mock-log-appender",
            RegexFilter.createFilter(filter, new String[0], false, null, null),
            Collections.unmodifiableList(Arrays.asList(loggers))
        );
        appender.start();
        for (Logger logger : loggers) {
            Loggers.addAppender(logger, appender);
        }
        return appender;
    }

    private MockLogAppender(String name, RegexFilter filter, List loggers) {
        super(name, filter, null, true, Property.EMPTY_ARRAY);
        /*
         * 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.
         */
        this.expectations = new CopyOnWriteArrayList<>();
        this.loggers = loggers;
    }

    public void addExpectation(LoggingExpectation expectation) {
        expectations.add(expectation);
    }

    @Override
    public void append(LogEvent event) {
        for (LoggingExpectation expectation : expectations) {
            expectation.match(event);
        }
    }

    public void assertAllExpectationsMatched() {
        for (LoggingExpectation expectation : expectations) {
            expectation.assertMatched();
        }
    }

    @Override
    public void close() {
        for (Logger logger : loggers) {
            Loggers.removeAppender(logger, this);
        }
        super.stop();
    }

    @Override
    public void stop() {
        // MockLogAppender should be used with try-with-resources to ensure
        // proper clean up ordering and should never be stopped directly.
        throw new UnsupportedOperationException("Use close() to ensure proper clean up ordering");
    }

    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 = getLoggerName(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 ExceptionSeenEventExpectation extends SeenEventExpectation {

        private final Class clazz;
        private final String exceptionMessage;

        public ExceptionSeenEventExpectation(
            final String name,
            final String logger,
            final Level level,
            final String message,
            final Class 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 String 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;
        }

        @Override
        public void match(LogEvent event) {
            if (event.getLevel().equals(level) && event.getLoggerName().equals(logger)) {
                if (Pattern.matches(pattern, event.getMessage().getFormattedMessage())) {
                    saw = true;
                }
            }
        }

        @Override
        public void assertMatched() {
            assertThat(name, saw, equalTo(true));
        }

    }

    /**
     * Used for cases when the logger is dynamically named such as to include an index name or shard id
     *
     * Best used in conjunction with the root logger:
     * {@code @TestLogging(value = "_root:debug", reason = "Validate logging output");}
     * @see TestLogging
     * */
    public static class PatternSeenWithLoggerPrefixExpectation implements LoggingExpectation {
        private final String expectationName;
        private final String loggerPrefix;
        private final Level level;
        private final String messageMatchingRegex;

        private final List loggerMatches = new ArrayList<>();
        private final AtomicBoolean eventSeen = new AtomicBoolean(false);

        public PatternSeenWithLoggerPrefixExpectation(
            final String expectationName,
            final String loggerPrefix,
            final Level level,
            final String messageMatchingRegex
        ) {
            this.expectationName = expectationName;
            this.loggerPrefix = loggerPrefix;
            this.level = level;
            this.messageMatchingRegex = messageMatchingRegex;
        }

        @Override
        public void match(final LogEvent event) {
            if (event.getLevel() == level && event.getLoggerName().startsWith(loggerPrefix)) {
                final String formattedMessage = event.getMessage().getFormattedMessage();
                loggerMatches.add(formattedMessage);
                if (formattedMessage.matches(messageMatchingRegex)) {
                    eventSeen.set(true);
                }
            }
        }

        @Override
        public void assertMatched() {
            if (!eventSeen.get()) {
                final StringBuilder failureMessage = new StringBuilder();
                failureMessage.append(expectationName + " was not seen, found " + loggerMatches.size() + " messages matching the logger.");
                failureMessage.append("\r\nMessage matching regex: " + messageMatchingRegex);
                if (!loggerMatches.isEmpty()) {
                    failureMessage.append("\r\nMessage details:\r\n" + String.join("\r\n", loggerMatches));
                }
                fail(failureMessage.toString());
            }
        }
    }

    private static String getLoggerName(String name) {
        if (name.startsWith("org.opensearch.")) {
            name = name.substring("org.opensearch.".length());
        }
        return COMMON_PREFIX + name;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy