Skip to content

Commit c7b019c

Browse files
committedJul 24, 2009
@required does not get processed on beans returned by @bean factory methods (SPR-5744)
1 parent 840ac88 commit c7b019c

File tree

3 files changed

+71
-15
lines changed

3 files changed

+71
-15
lines changed
 

‎org.springframework.beans/src/main/java/org/springframework/beans/factory/annotation/RequiredAnnotationBeanPostProcessor.java

+46-9
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2007 the original author or authors.
2+
* Copyright 2002-2009 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -27,8 +27,12 @@
2727

2828
import org.springframework.beans.BeansException;
2929
import org.springframework.beans.PropertyValues;
30+
import org.springframework.beans.factory.BeanFactory;
31+
import org.springframework.beans.factory.BeanFactoryAware;
3032
import org.springframework.beans.factory.BeanInitializationException;
33+
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
3134
import org.springframework.beans.factory.config.InstantiationAwareBeanPostProcessorAdapter;
35+
import org.springframework.core.Conventions;
3236
import org.springframework.core.Ordered;
3337
import org.springframework.core.PriorityOrdered;
3438
import org.springframework.core.annotation.AnnotationUtils;
@@ -66,12 +70,23 @@
6670
* @see Required
6771
*/
6872
public class RequiredAnnotationBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter
69-
implements PriorityOrdered {
73+
implements PriorityOrdered, BeanFactoryAware {
74+
75+
/**
76+
* Bean definition attribute that may indicate whether a given bean is supposed
77+
* to be skipped when performing this post-processor's required property check.
78+
* @see #shouldSkip
79+
*/
80+
public static final String SKIP_REQUIRED_CHECK_ATTRIBUTE =
81+
Conventions.getQualifiedAttributeName(RequiredAnnotationBeanPostProcessor.class, "skipRequiredCheck");
82+
7083

7184
private Class<? extends Annotation> requiredAnnotationType = Required.class;
7285

7386
private int order = Ordered.LOWEST_PRECEDENCE - 1;
7487

88+
private ConfigurableListableBeanFactory beanFactory;
89+
7590
/** Cache for validated bean names, skipping re-validation for the same bean */
7691
private final Set<String> validatedBeanNames = Collections.synchronizedSet(new HashSet<String>());
7792

@@ -97,6 +112,12 @@ protected Class<? extends Annotation> getRequiredAnnotationType() {
97112
return this.requiredAnnotationType;
98113
}
99114

115+
public void setBeanFactory(BeanFactory beanFactory) {
116+
if (beanFactory instanceof ConfigurableListableBeanFactory) {
117+
this.beanFactory = (ConfigurableListableBeanFactory) beanFactory;
118+
}
119+
}
120+
100121
public void setOrder(int order) {
101122
this.order = order;
102123
}
@@ -112,20 +133,36 @@ public PropertyValues postProcessPropertyValues(
112133
throws BeansException {
113134

114135
if (!this.validatedBeanNames.contains(beanName)) {
115-
List<String> invalidProperties = new ArrayList<String>();
116-
for (PropertyDescriptor pd : pds) {
117-
if (isRequiredProperty(pd) && !pvs.contains(pd.getName())) {
118-
invalidProperties.add(pd.getName());
136+
if (!shouldSkip(this.beanFactory, beanName)) {
137+
List<String> invalidProperties = new ArrayList<String>();
138+
for (PropertyDescriptor pd : pds) {
139+
if (isRequiredProperty(pd) && !pvs.contains(pd.getName())) {
140+
invalidProperties.add(pd.getName());
141+
}
142+
}
143+
if (!invalidProperties.isEmpty()) {
144+
throw new BeanInitializationException(buildExceptionMessage(invalidProperties, beanName));
119145
}
120-
}
121-
if (!invalidProperties.isEmpty()) {
122-
throw new BeanInitializationException(buildExceptionMessage(invalidProperties, beanName));
123146
}
124147
this.validatedBeanNames.add(beanName);
125148
}
126149
return pvs;
127150
}
128151

152+
/**
153+
* Check whether the given bean definition is not subject to the annotation-based
154+
* required property check as performed by this post-processor.
155+
* <p>The default implementations check for the presence of the
156+
* {@link #SKIP_REQUIRED_CHECK_ATTRIBUTE} attribute in the bean definition, if any.
157+
* @param beanFactory the BeanFactory to check against
158+
* @param beanName the name of the bean to check against
159+
* @return <code>true</code> to skip the bean; <code>false</code> to process it
160+
*/
161+
protected boolean shouldSkip(ConfigurableListableBeanFactory beanFactory, String beanName) {
162+
return (beanFactory != null && beanFactory.containsBeanDefinition(beanName) &&
163+
Boolean.TRUE.equals(beanFactory.getBeanDefinition(beanName).getAttribute(SKIP_REQUIRED_CHECK_ATTRIBUTE)));
164+
}
165+
129166
/**
130167
* Is the supplied property required to have a value (that is, to be dependency-injected)?
131168
* <p>This implementation looks for the existence of a

‎org.springframework.context/src/main/java/org/springframework/context/annotation/ConfigurationClassBeanDefinitionReader.java

+2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828

2929
import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
3030
import org.springframework.beans.factory.annotation.Autowire;
31+
import org.springframework.beans.factory.annotation.RequiredAnnotationBeanPostProcessor;
3132
import org.springframework.beans.factory.config.BeanDefinition;
3233
import org.springframework.beans.factory.config.BeanDefinitionHolder;
3334
import org.springframework.beans.factory.parsing.SourceExtractor;
@@ -123,6 +124,7 @@ private void loadBeanDefinitionsForModelMethod(ConfigurationClassMethod method)
123124
beanDef.setFactoryBeanName(configClass.getBeanName());
124125
beanDef.setUniqueFactoryMethodName(metadata.getMethodName());
125126
beanDef.setAutowireMode(RootBeanDefinition.AUTOWIRE_CONSTRUCTOR);
127+
beanDef.setAttribute(RequiredAnnotationBeanPostProcessor.SKIP_REQUIRED_CHECK_ATTRIBUTE, Boolean.TRUE);
126128

127129
// consider name and any aliases
128130
Map<String, Object> beanAttributes = metadata.getAnnotationAttributes(Bean.class.getName());

‎org.springframework.context/src/test/java/org/springframework/context/annotation/configuration/ConfigurationClassProcessingTests.java

+23-6
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,8 @@
2323

2424
import org.springframework.beans.factory.BeanFactory;
2525
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
26-
import org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor;
26+
import org.springframework.beans.factory.annotation.Required;
27+
import org.springframework.beans.factory.annotation.RequiredAnnotationBeanPostProcessor;
2728
import org.springframework.beans.factory.config.BeanDefinition;
2829
import org.springframework.beans.factory.parsing.BeanDefinitionParsingException;
2930
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
@@ -56,9 +57,11 @@ private BeanFactory initBeanFactory(Class<?>... configClasses) {
5657
String configBeanName = configClass.getName();
5758
factory.registerBeanDefinition(configBeanName, new RootBeanDefinition(configClass));
5859
}
59-
ConfigurationClassPostProcessor pp = new ConfigurationClassPostProcessor();
60-
pp.postProcessBeanFactory(factory);
61-
factory.addBeanPostProcessor(new AutowiredAnnotationBeanPostProcessor());
60+
ConfigurationClassPostProcessor ccpp = new ConfigurationClassPostProcessor();
61+
ccpp.postProcessBeanFactory(factory);
62+
RequiredAnnotationBeanPostProcessor rapp = new RequiredAnnotationBeanPostProcessor();
63+
rapp.setBeanFactory(factory);
64+
factory.addBeanPostProcessor(rapp);
6265
return factory;
6366
}
6467

@@ -161,13 +164,13 @@ public TestBean methodName() {
161164
static class ConfigWithPrototypeBean {
162165

163166
public @Bean TestBean foo() {
164-
TestBean foo = new TestBean("foo");
167+
TestBean foo = new SpousyTestBean("foo");
165168
foo.setSpouse(bar());
166169
return foo;
167170
}
168171

169172
public @Bean TestBean bar() {
170-
TestBean bar = new TestBean("bar");
173+
TestBean bar = new SpousyTestBean("bar");
171174
bar.setSpouse(baz());
172175
return bar;
173176
}
@@ -178,4 +181,18 @@ public TestBean baz() {
178181
}
179182
}
180183

184+
185+
private static class SpousyTestBean extends TestBean {
186+
187+
public SpousyTestBean(String name) {
188+
super(name);
189+
}
190+
191+
@Override
192+
@Required
193+
public void setSpouse(ITestBean spouse) {
194+
super.setSpouse(spouse);
195+
}
196+
}
197+
181198
}

0 commit comments

Comments
 (0)
Please sign in to comment.