/* * Copyright 2012-2017 the original author or authors. * * Licensed 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.springframework.boot.autoconfigure.logging; import java.util.Arrays; import ch.qos.logback.classic.Level; import ch.qos.logback.classic.Logger; import ch.qos.logback.classic.LoggerContext; import org.junit.Rule; import org.junit.Test; import org.slf4j.impl.StaticLoggerBinder; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport; import org.springframework.boot.autoconfigure.context.PropertyPlaceholderAutoConfiguration; import org.springframework.boot.autoconfigure.http.HttpMessageConvertersAutoConfiguration; import org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration; import org.springframework.boot.context.event.ApplicationFailedEvent; import org.springframework.boot.testutil.InternalOutputCapture; import org.springframework.context.annotation.AnnotationConfigApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; import org.springframework.context.event.ContextRefreshedEvent; import org.springframework.mock.web.MockServletContext; import org.springframework.web.context.support.AnnotationConfigWebApplicationContext; import static org.assertj.core.api.Assertions.assertThat; import static org.junit.Assert.fail; /** * Tests for {@link AutoConfigurationReportLoggingInitializer}. * * @author Phillip Webb * @author Andy Wilkinson */ public class AutoConfigurationReportLoggingInitializerTests { @Rule public InternalOutputCapture outputCapture = new InternalOutputCapture(); private AutoConfigurationReportLoggingInitializer initializer = new AutoConfigurationReportLoggingInitializer(); @Test public void logsDebugOnContextRefresh() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); this.initializer.initialize(context); context.register(Config.class); context.refresh(); withDebugLogging(() -> { this.initializer.onApplicationEvent(new ContextRefreshedEvent(context)); }); assertThat(this.outputCapture.toString()).contains("AUTO-CONFIGURATION REPORT"); } @Test public void logsDebugOnError() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); this.initializer.initialize(context); context.register(ErrorConfig.class); try { context.refresh(); fail("Did not error"); } catch (Exception ex) { withDebugLogging(() -> { this.initializer.onApplicationEvent(new ApplicationFailedEvent( new SpringApplication(), new String[0], context, ex)); }); } assertThat(this.outputCapture.toString()).contains("AUTO-CONFIGURATION REPORT"); } @Test public void logsInfoOnErrorIfDebugDisabled() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); this.initializer.initialize(context); context.register(ErrorConfig.class); try { context.refresh(); fail("Did not error"); } catch (Exception ex) { this.initializer.onApplicationEvent(new ApplicationFailedEvent( new SpringApplication(), new String[0], context, ex)); } assertThat(this.outputCapture.toString()).contains("Error starting" + " ApplicationContext. To display the auto-configuration report re-run" + " your application with 'debug' enabled."); } @Test public void logsOutput() throws Exception { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); this.initializer.initialize(context); context.register(Config.class); ConditionEvaluationReport.get(context.getBeanFactory()) .recordExclusions(Arrays.asList("com.foo.Bar")); context.refresh(); withDebugLogging(() -> { this.initializer.onApplicationEvent(new ContextRefreshedEvent(context)); }); assertThat(this.outputCapture.toString()) .contains("not a servlet web application (OnWebApplicationCondition)"); } @Test public void canBeUsedInApplicationContext() throws Exception { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(Config.class); new AutoConfigurationReportLoggingInitializer().initialize(context); context.refresh(); assertThat(context.getBean(ConditionEvaluationReport.class)).isNotNull(); } @Test public void canBeUsedInNonGenericApplicationContext() throws Exception { AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext(); context.setServletContext(new MockServletContext()); context.register(Config.class); new AutoConfigurationReportLoggingInitializer().initialize(context); context.refresh(); assertThat(context.getBean(ConditionEvaluationReport.class)).isNotNull(); } @Test public void noErrorIfNotInitialized() throws Exception { this.initializer .onApplicationEvent(new ApplicationFailedEvent(new SpringApplication(), new String[0], null, new RuntimeException("Planned"))); assertThat(this.outputCapture.toString()) .contains("Unable to provide auto-configuration report"); } private void withDebugLogging(Runnable runnable) { LoggerContext context = (LoggerContext) StaticLoggerBinder.getSingleton() .getLoggerFactory(); Logger logger = context .getLogger(AutoConfigurationReportLoggingInitializer.class); Level currentLevel = logger.getLevel(); logger.setLevel(Level.DEBUG); try { runnable.run(); } finally { logger.setLevel(currentLevel); } } @Configuration @Import({ WebMvcAutoConfiguration.class, HttpMessageConvertersAutoConfiguration.class, PropertyPlaceholderAutoConfiguration.class }) static class Config { } @Configuration @Import(WebMvcAutoConfiguration.class) static class ErrorConfig { @Bean public String iBreak() { throw new RuntimeException(); } } }