blob: e4e21c76f376c20e8ea0b6eff0251fc8df3c2bf1 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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.
*/
package org.apache.logging.log4j.web;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertSame;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
import static org.mockito.ArgumentMatchers.eq;
import static org.mockito.BDDMockito.any;
import static org.mockito.BDDMockito.given;
import static org.mockito.BDDMockito.never;
import static org.mockito.BDDMockito.then;
import static org.mockito.BDDMockito.willThrow;
import static org.mockito.Mockito.mock;
import jakarta.servlet.DispatcherType;
import jakarta.servlet.Filter;
import jakarta.servlet.FilterRegistration;
import jakarta.servlet.ServletContainerInitializer;
import jakarta.servlet.ServletContext;
import java.util.EnumSet;
import java.util.EventListener;
import java.util.ServiceLoader;
import java.util.stream.StreamSupport;
import org.apache.logging.log4j.util.Strings;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.ArgumentCaptor;
import org.mockito.Captor;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension;
@ExtendWith(MockitoExtension.class)
public class Log4jServletContainerInitializerTest {
@Mock
private ServletContext servletContext;
@Mock
private Log4jWebLifeCycle initializer;
@Captor
private ArgumentCaptor<Class<? extends Filter>> filterCaptor;
@Captor
private ArgumentCaptor<EventListener> listenerCaptor;
private Log4jServletContainerInitializer containerInitializer;
@BeforeEach
public void setUp() {
this.containerInitializer = new Log4jServletContainerInitializer();
}
@Test
public void testOnStartupWithServletVersion2_x() throws Exception {
given(servletContext.getMajorVersion()).willReturn(2);
this.containerInitializer.onStartup(null, this.servletContext);
}
@Test
public void testOnStartupWithServletVersion3_xEffectiveVersion2_x() throws Exception {
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(2);
this.containerInitializer.onStartup(null, this.servletContext);
}
@Test
public void testOnStartupWithServletVersion3_xEffectiveVersion3_xDisabledTrue() throws Exception {
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn("true");
this.containerInitializer.onStartup(null, this.servletContext);
}
@Test
public void testOnStartupWithServletVersion3_xEffectiveVersion3_xShutdownDisabled() throws Exception {
final FilterRegistration.Dynamic registration = mock(FilterRegistration.Dynamic.class);
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_SHUTDOWN_DISABLED)))
.willReturn("true");
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn(null);
given(servletContext.addFilter(eq("log4jServletFilter"), filterCaptor.capture()))
.willReturn(registration);
given(servletContext.getAttribute(Log4jWebSupport.SUPPORT_ATTRIBUTE)).willReturn(initializer);
this.containerInitializer.onStartup(null, this.servletContext);
then(initializer).should().start();
then(initializer).should().setLoggerContext();
then(registration).should().setAsyncSupported(eq(true));
then(registration)
.should()
.addMappingForUrlPatterns(eq(EnumSet.allOf(DispatcherType.class)), eq(false), eq("/*"));
// initParam IS_LOG4J_AUTO_SHUTDOWN_DISABLED is "true" so addListener shouldn't be called.
then(servletContext).should(never()).addListener(any(Log4jServletContextListener.class));
}
@Test
public void testOnStartupWithServletVersion3_xEffectiveVersion3_xDisabledTRUE() throws Exception {
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn("TRUE");
this.containerInitializer.onStartup(null, this.servletContext);
}
@Test
public void testOnStartupWithServletVersion3_xEffectiveVersion3_x() throws Exception {
final FilterRegistration.Dynamic registration = mock(FilterRegistration.Dynamic.class);
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn(null);
given(servletContext.addFilter(eq("log4jServletFilter"), filterCaptor.capture()))
.willReturn(registration);
given(servletContext.getAttribute(Log4jWebSupport.SUPPORT_ATTRIBUTE)).willReturn(initializer);
containerInitializer.onStartup(null, servletContext);
then(initializer).should().start();
then(initializer).should().setLoggerContext();
then(servletContext).should().addListener(listenerCaptor.capture());
then(registration).should().setAsyncSupported(eq(true));
then(registration)
.should()
.addMappingForUrlPatterns(eq(EnumSet.allOf(DispatcherType.class)), eq(false), eq("/*"));
assertNotNull(listenerCaptor.getValue(), "The listener should not be null.");
assertSame(
Log4jServletContextListener.class,
listenerCaptor.getValue().getClass(),
"The listener is not correct.");
assertNotNull(filterCaptor.getValue(), "The filter should not be null.");
assertSame(Log4jServletFilter.class, filterCaptor.getValue(), "The filter is not correct.");
}
@Test
public void testOnStartupCanceledDueToPreExistingFilter() throws Exception {
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn("false");
given(servletContext.addFilter(eq("log4jServletFilter"), filterCaptor.capture()))
.willReturn(null);
this.containerInitializer.onStartup(null, this.servletContext);
assertNotNull(filterCaptor.getValue(), "The filter should not be null.");
assertSame(Log4jServletFilter.class, filterCaptor.getValue(), "The filter is not correct.");
}
@Test
public void testOnStartupFailedDueToInitializerFailure() throws Exception {
final FilterRegistration.Dynamic registration = mock(FilterRegistration.Dynamic.class);
final IllegalStateException exception = new IllegalStateException(Strings.EMPTY);
given(servletContext.getMajorVersion()).willReturn(3);
given(servletContext.getEffectiveMajorVersion()).willReturn(3);
given(servletContext.getInitParameter(eq(Log4jWebSupport.IS_LOG4J_AUTO_INITIALIZATION_DISABLED)))
.willReturn("balderdash");
given(servletContext.addFilter(eq("log4jServletFilter"), filterCaptor.capture()))
.willReturn(registration);
given(servletContext.getAttribute(Log4jWebSupport.SUPPORT_ATTRIBUTE)).willReturn(initializer);
willThrow(exception).given(initializer).start();
try {
this.containerInitializer.onStartup(null, this.servletContext);
fail("Expected the exception thrown by the initializer; got no exception.");
} catch (final IllegalStateException e) {
assertSame(exception, e, "The exception is not correct.");
}
then(initializer).should().start();
assertNotNull(filterCaptor.getValue(), "The filter should not be null.");
assertSame(Log4jServletFilter.class, filterCaptor.getValue(), "The filter is not correct.");
}
@Test
public void testServiceIsDetected() {
final ServiceLoader<ServletContainerInitializer> loader = ServiceLoader.load(ServletContainerInitializer.class);
final boolean found = StreamSupport.stream(loader.spliterator(), false)
.anyMatch(s -> s instanceof Log4jServletContainerInitializer);
assertTrue(found, "ServletContainerInitializer not found.");
}
}