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

org.springframework.boot.diagnostics.FailureAnalyzers Maven / Gradle / Ivy

/*
 * Copyright 2012-2023 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
 *
 *      https://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.diagnostics;

import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.boot.SpringBootExceptionReporter;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.EnvironmentAware;
import org.springframework.core.env.Environment;
import org.springframework.core.io.support.SpringFactoriesLoader;
import org.springframework.core.io.support.SpringFactoriesLoader.ArgumentResolver;
import org.springframework.core.io.support.SpringFactoriesLoader.FailureHandler;
import org.springframework.core.log.LogMessage;
import org.springframework.util.StringUtils;

/**
 * Utility to trigger {@link FailureAnalyzer} and {@link FailureAnalysisReporter}
 * instances loaded from {@code spring.factories}.
 * 

* A {@code FailureAnalyzer} that requires access to the {@link BeanFactory} or * {@link Environment} in order to perform its analysis can implement a constructor that * accepts arguments of one or both of these types. * * @author Andy Wilkinson * @author Phillip Webb * @author Stephane Nicoll * @author Scott Frederick */ final class FailureAnalyzers implements SpringBootExceptionReporter { private static final Log logger = LogFactory.getLog(FailureAnalyzers.class); private final SpringFactoriesLoader springFactoriesLoader; private final List analyzers; public FailureAnalyzers(ConfigurableApplicationContext context) { this(context, SpringFactoriesLoader.forDefaultResourceLocation((context != null) ? context.getClassLoader() : null)); } FailureAnalyzers(ConfigurableApplicationContext context, SpringFactoriesLoader springFactoriesLoader) { this.springFactoriesLoader = springFactoriesLoader; this.analyzers = loadFailureAnalyzers(context, this.springFactoriesLoader); } private static List loadFailureAnalyzers(ConfigurableApplicationContext context, SpringFactoriesLoader springFactoriesLoader) { List analyzers = springFactoriesLoader.load(FailureAnalyzer.class, getArgumentResolver(context), FailureHandler.logging(logger)); List awareAnalyzers = analyzers.stream() .filter((analyzer) -> analyzer instanceof BeanFactoryAware || analyzer instanceof EnvironmentAware) .toList(); if (!awareAnalyzers.isEmpty()) { String awareAnalyzerNames = StringUtils.collectionToCommaDelimitedString( awareAnalyzers.stream().map((analyzer) -> analyzer.getClass().getName()).toList()); logger.warn(LogMessage.format( "FailureAnalyzers [%s] implement BeanFactoryAware or EnvironmentAware. " + "Support for these interfaces on FailureAnalyzers is deprecated, " + "and will be removed in a future release. " + "Instead provide a constructor that accepts BeanFactory or Environment parameters.", awareAnalyzerNames)); if (context == null) { logger.trace(LogMessage.format("Skipping [%s] due to missing context", awareAnalyzerNames)); return analyzers.stream().filter((analyzer) -> !awareAnalyzers.contains(analyzer)).toList(); } awareAnalyzers.forEach((analyzer) -> { if (analyzer instanceof BeanFactoryAware beanFactoryAware) { beanFactoryAware.setBeanFactory(context.getBeanFactory()); } if (analyzer instanceof EnvironmentAware environmentAware) { environmentAware.setEnvironment(context.getEnvironment()); } }); } return analyzers; } private static ArgumentResolver getArgumentResolver(ConfigurableApplicationContext context) { if (context == null) { return null; } ArgumentResolver argumentResolver = ArgumentResolver.of(BeanFactory.class, context.getBeanFactory()); argumentResolver = argumentResolver.and(Environment.class, context.getEnvironment()); return argumentResolver; } @Override public boolean reportException(Throwable failure) { FailureAnalysis analysis = analyze(failure, this.analyzers); return report(analysis); } private FailureAnalysis analyze(Throwable failure, List analyzers) { for (FailureAnalyzer analyzer : analyzers) { try { FailureAnalysis analysis = analyzer.analyze(failure); if (analysis != null) { return analysis; } } catch (Throwable ex) { logger.trace(LogMessage.format("FailureAnalyzer %s failed", analyzer), ex); } } return null; } private boolean report(FailureAnalysis analysis) { List reporters = this.springFactoriesLoader.load(FailureAnalysisReporter.class); if (analysis == null || reporters.isEmpty()) { return false; } for (FailureAnalysisReporter reporter : reporters) { reporter.report(analysis); } return true; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy