Skip to content

Commit 07d0794

Browse files
committed
Consider possibility of missing @ConstructorBinding in failure analysis
Previously, when a NoSuchBeanDefinitionException was being analyzed, the possibility of a missing @ConstructorBinding annotation on a @ConfigurationProperties class was not considered. This commit updates the failure analysis for failed constructor injection of an instance of a @ConfigurationProperties-annotated class. When such a failure occurs, adding @ConstructorBinding is now suggested as an action. Closes gh-18545
1 parent 04e035c commit 07d0794

File tree

2 files changed

+63
-8
lines changed

2 files changed

+63
-8
lines changed

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzer.java

Lines changed: 29 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
package org.springframework.boot.autoconfigure.diagnostics.analyzer;
1818

1919
import java.lang.annotation.Annotation;
20+
import java.lang.reflect.Constructor;
2021
import java.util.ArrayList;
2122
import java.util.Arrays;
2223
import java.util.Collections;
@@ -30,6 +31,7 @@
3031
import org.springframework.beans.factory.BeanFactory;
3132
import org.springframework.beans.factory.BeanFactoryAware;
3233
import org.springframework.beans.factory.BeanFactoryUtils;
34+
import org.springframework.beans.factory.InjectionPoint;
3335
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
3436
import org.springframework.beans.factory.UnsatisfiedDependencyException;
3537
import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
@@ -39,10 +41,14 @@
3941
import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcome;
4042
import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcomes;
4143
import org.springframework.boot.autoconfigure.condition.ConditionOutcome;
44+
import org.springframework.boot.context.properties.ConfigurationProperties;
45+
import org.springframework.boot.context.properties.ConstructorBinding;
4246
import org.springframework.boot.diagnostics.FailureAnalysis;
4347
import org.springframework.boot.diagnostics.analyzer.AbstractInjectionFailureAnalyzer;
4448
import org.springframework.context.annotation.Bean;
4549
import org.springframework.core.ResolvableType;
50+
import org.springframework.core.annotation.MergedAnnotation;
51+
import org.springframework.core.annotation.MergedAnnotations;
4652
import org.springframework.core.type.MethodMetadata;
4753
import org.springframework.core.type.classreading.CachingMetadataReaderFactory;
4854
import org.springframework.core.type.classreading.MetadataReader;
@@ -85,11 +91,14 @@ protected FailureAnalysis analyze(Throwable rootFailure, NoSuchBeanDefinitionExc
8591
StringBuilder message = new StringBuilder();
8692
message.append(String.format("%s required %s that could not be found.%n",
8793
(description != null) ? description : "A component", getBeanDescription(cause)));
88-
List<Annotation> injectionAnnotations = findInjectionAnnotations(rootFailure);
89-
if (!injectionAnnotations.isEmpty()) {
90-
message.append(String.format("%nThe injection point has the following annotations:%n"));
91-
for (Annotation injectionAnnotation : injectionAnnotations) {
92-
message.append(String.format("\t- %s%n", injectionAnnotation));
94+
InjectionPoint injectionPoint = findInjectionPoint(rootFailure);
95+
if (injectionPoint != null) {
96+
Annotation[] injectionAnnotations = injectionPoint.getAnnotations();
97+
if (injectionAnnotations.length > 0) {
98+
message.append(String.format("%nThe injection point has the following annotations:%n"));
99+
for (Annotation injectionAnnotation : injectionAnnotations) {
100+
message.append(String.format("\t- %s%n", injectionAnnotation));
101+
}
93102
}
94103
}
95104
if (!autoConfigurationResults.isEmpty() || !userConfigurationResults.isEmpty()) {
@@ -105,6 +114,18 @@ protected FailureAnalysis analyze(Throwable rootFailure, NoSuchBeanDefinitionExc
105114
(!autoConfigurationResults.isEmpty() || !userConfigurationResults.isEmpty())
106115
? "revisiting the entries above or defining" : "defining",
107116
getBeanDescription(cause));
117+
if (injectionPoint != null && injectionPoint.getMember() instanceof Constructor) {
118+
Constructor<?> constructor = (Constructor<?>) injectionPoint.getMember();
119+
Class<?> declaringClass = constructor.getDeclaringClass();
120+
MergedAnnotation<ConfigurationProperties> configurationProperties = MergedAnnotations.from(declaringClass)
121+
.get(ConfigurationProperties.class);
122+
if (configurationProperties.isPresent()) {
123+
action = String.format(
124+
"%s%nConsider adding @%s to %s if you intended to use constructor-based "
125+
+ "configuration property binding.",
126+
action, ConstructorBinding.class.getSimpleName(), constructor.getName());
127+
}
128+
}
108129
return new FailureAnalysis(message.toString(), action, cause);
109130
}
110131

@@ -182,13 +203,13 @@ private void collectExcludedAutoConfiguration(NoSuchBeanDefinitionException caus
182203
}
183204
}
184205

185-
private List<Annotation> findInjectionAnnotations(Throwable failure) {
206+
private InjectionPoint findInjectionPoint(Throwable failure) {
186207
UnsatisfiedDependencyException unsatisfiedDependencyException = findCause(failure,
187208
UnsatisfiedDependencyException.class);
188209
if (unsatisfiedDependencyException == null) {
189-
return Collections.emptyList();
210+
return null;
190211
}
191-
return Arrays.asList(unsatisfiedDependencyException.getInjectionPoint().getAnnotations());
212+
return unsatisfiedDependencyException.getInjectionPoint();
192213
}
193214

194215
private class Source {

spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzerTests.java

Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,8 @@
3030
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
3131
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
3232
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
33+
import org.springframework.boot.context.properties.ConfigurationProperties;
34+
import org.springframework.boot.context.properties.EnableConfigurationProperties;
3335
import org.springframework.boot.diagnostics.FailureAnalysis;
3436
import org.springframework.boot.diagnostics.LoggingFailureAnalysisReporter;
3537
import org.springframework.boot.test.util.TestPropertyValues;
@@ -155,6 +157,16 @@ void failureAnalysisForUnmatchedQualifier() {
155157
.containsPattern("@org.springframework.beans.factory.annotation.Qualifier\\(value=\"*alpha\"*\\)");
156158
}
157159

160+
@Test
161+
void failureAnalysisForConfigurationPropertiesThatMaybeShouldHaveBeenConstructorBound() {
162+
FailureAnalysis analysis = analyzeFailure(
163+
createFailure(ConstructorBoundConfigurationPropertiesConfiguration.class));
164+
assertThat(analysis.getAction()).startsWith(
165+
String.format("Consider defining a bean of type '%s' in your configuration.", String.class.getName()));
166+
assertThat(analysis.getAction()).contains(
167+
"Consider adding @ConstructorBinding to " + NeedsConstructorBindingProperties.class.getName());
168+
}
169+
158170
private void assertDescriptionConstructorMissingType(FailureAnalysis analysis, Class<?> component, int index,
159171
Class<?> type) {
160172
String expected = String.format(
@@ -167,6 +179,7 @@ private void assertActionMissingType(FailureAnalysis analysis, Class<?> type) {
167179
assertThat(analysis.getAction()).startsWith(String.format(
168180
"Consider revisiting the entries above or defining a bean of type '%s' in your configuration.",
169181
type.getName()));
182+
assertThat(analysis.getAction()).doesNotContain("@ConstructorBinding");
170183
}
171184

172185
private void assertActionMissingName(FailureAnalysis analysis, String name) {
@@ -359,4 +372,25 @@ static class StringNameHandler {
359372

360373
}
361374

375+
@Configuration(proxyBeanMethods = false)
376+
@EnableConfigurationProperties(NeedsConstructorBindingProperties.class)
377+
static class ConstructorBoundConfigurationPropertiesConfiguration {
378+
379+
}
380+
381+
@ConfigurationProperties("test")
382+
static class NeedsConstructorBindingProperties {
383+
384+
private final String name;
385+
386+
NeedsConstructorBindingProperties(String name) {
387+
this.name = name;
388+
}
389+
390+
String getName() {
391+
return this.name;
392+
}
393+
394+
}
395+
362396
}

0 commit comments

Comments
 (0)