Expose a way to resolved a merged inner bean definition
This commit extracts the logic of resolving a merged bean definition for an inner bean to a public method so that other components can reuse it. Closes gh-28093
This commit is contained in:
parent
fd191d165b
commit
cc57b55c61
|
@ -23,6 +23,7 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.function.BiFunction;
|
||||||
|
|
||||||
import org.springframework.beans.BeanWrapper;
|
import org.springframework.beans.BeanWrapper;
|
||||||
import org.springframework.beans.BeanWrapperImpl;
|
import org.springframework.beans.BeanWrapperImpl;
|
||||||
|
@ -57,6 +58,7 @@ import org.springframework.util.StringUtils;
|
||||||
*
|
*
|
||||||
* @author Juergen Hoeller
|
* @author Juergen Hoeller
|
||||||
* @author Sam Brannen
|
* @author Sam Brannen
|
||||||
|
* @author Stephane Nicoll
|
||||||
* @since 1.2
|
* @since 1.2
|
||||||
* @see AbstractAutowireCapableBeanFactory
|
* @see AbstractAutowireCapableBeanFactory
|
||||||
*/
|
*/
|
||||||
|
@ -143,13 +145,12 @@ public class BeanDefinitionValueResolver {
|
||||||
}
|
}
|
||||||
else if (value instanceof BeanDefinitionHolder bdHolder) {
|
else if (value instanceof BeanDefinitionHolder bdHolder) {
|
||||||
// Resolve BeanDefinitionHolder: contains BeanDefinition with name and aliases.
|
// Resolve BeanDefinitionHolder: contains BeanDefinition with name and aliases.
|
||||||
return resolveInnerBean(argName, bdHolder.getBeanName(), bdHolder.getBeanDefinition());
|
return resolveInnerBean(bdHolder.getBeanName(), bdHolder.getBeanDefinition(),
|
||||||
|
(name, mbd) -> resolveInnerBeanValue(argName, name, mbd));
|
||||||
}
|
}
|
||||||
else if (value instanceof BeanDefinition bd) {
|
else if (value instanceof BeanDefinition bd) {
|
||||||
// Resolve plain BeanDefinition, without contained name: use dummy name.
|
return resolveInnerBean(null, bd,
|
||||||
String innerBeanName = "(inner bean)" + BeanFactoryUtils.GENERATED_BEAN_NAME_SEPARATOR +
|
(name, mbd) -> resolveInnerBeanValue(argName, name, mbd));
|
||||||
ObjectUtils.getIdentityHexString(bd);
|
|
||||||
return resolveInnerBean(argName, innerBeanName, bd);
|
|
||||||
}
|
}
|
||||||
else if (value instanceof DependencyDescriptor dependencyDescriptor) {
|
else if (value instanceof DependencyDescriptor dependencyDescriptor) {
|
||||||
Set<String> autowiredBeanNames = new LinkedHashSet<>(4);
|
Set<String> autowiredBeanNames = new LinkedHashSet<>(4);
|
||||||
|
@ -243,6 +244,23 @@ public class BeanDefinitionValueResolver {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve an inner bean definition and invoke the specified {@code resolver}
|
||||||
|
* on its merged bean definition.
|
||||||
|
* @param innerBeanName the inner bean name (or {@code null} to assign one)
|
||||||
|
* @param innerBd the inner raw bean definition
|
||||||
|
* @param resolver the function to invoke to resolve
|
||||||
|
* @param <T> the type of the resolution
|
||||||
|
* @return a resolved inner bean, as a result of applying the {@code resolver}
|
||||||
|
*/
|
||||||
|
public <T> T resolveInnerBean(@Nullable String innerBeanName, BeanDefinition innerBd,
|
||||||
|
BiFunction<String, RootBeanDefinition, T> resolver) {
|
||||||
|
String nameToUse = (innerBeanName != null ? innerBeanName : "(inner bean)"
|
||||||
|
+ BeanFactoryUtils.GENERATED_BEAN_NAME_SEPARATOR + ObjectUtils.getIdentityHexString(innerBd));
|
||||||
|
return resolver.apply(nameToUse, this.beanFactory.getMergedBeanDefinition(
|
||||||
|
nameToUse, innerBd, this.beanDefinition));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Evaluate the given value as an expression, if necessary.
|
* Evaluate the given value as an expression, if necessary.
|
||||||
* @param value the candidate value (may be an expression)
|
* @param value the candidate value (may be an expression)
|
||||||
|
@ -362,14 +380,12 @@ public class BeanDefinitionValueResolver {
|
||||||
* Resolve an inner bean definition.
|
* Resolve an inner bean definition.
|
||||||
* @param argName the name of the argument that the inner bean is defined for
|
* @param argName the name of the argument that the inner bean is defined for
|
||||||
* @param innerBeanName the name of the inner bean
|
* @param innerBeanName the name of the inner bean
|
||||||
* @param innerBd the bean definition for the inner bean
|
* @param mbd the merged bean definition for the inner bean
|
||||||
* @return the resolved inner bean instance
|
* @return the resolved inner bean instance
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
private Object resolveInnerBean(Object argName, String innerBeanName, BeanDefinition innerBd) {
|
private Object resolveInnerBeanValue(Object argName, String innerBeanName, RootBeanDefinition mbd) {
|
||||||
RootBeanDefinition mbd = null;
|
|
||||||
try {
|
try {
|
||||||
mbd = this.beanFactory.getMergedBeanDefinition(innerBeanName, innerBd, this.beanDefinition);
|
|
||||||
// Check given bean name whether it is unique. If not already unique,
|
// Check given bean name whether it is unique. If not already unique,
|
||||||
// add counter - increasing the counter until the name is unique.
|
// add counter - increasing the counter until the name is unique.
|
||||||
String actualInnerBeanName = innerBeanName;
|
String actualInnerBeanName = innerBeanName;
|
||||||
|
@ -400,7 +416,7 @@ public class BeanDefinitionValueResolver {
|
||||||
throw new BeanCreationException(
|
throw new BeanCreationException(
|
||||||
this.beanDefinition.getResourceDescription(), this.beanName,
|
this.beanDefinition.getResourceDescription(), this.beanName,
|
||||||
"Cannot create inner bean '" + innerBeanName + "' " +
|
"Cannot create inner bean '" + innerBeanName + "' " +
|
||||||
(mbd != null && mbd.getBeanClassName() != null ? "of type [" + mbd.getBeanClassName() + "] " : "") +
|
(mbd.getBeanClassName() != null ? "of type [" + mbd.getBeanClassName() + "] " : "") +
|
||||||
"while setting " + argName, ex);
|
"while setting " + argName, ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2022 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.beans.factory.support;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link BeanDefinitionValueResolver}.
|
||||||
|
*
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
*/
|
||||||
|
class BeanDefinitionValueResolverTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void resolveInnerBean() {
|
||||||
|
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();
|
||||||
|
RootBeanDefinition parentBd = new RootBeanDefinition();
|
||||||
|
GenericBeanDefinition innerBd = new GenericBeanDefinition();
|
||||||
|
innerBd.setAttribute("test", 42);
|
||||||
|
BeanDefinitionValueResolver bdvr = new BeanDefinitionValueResolver(beanFactory, "test", parentBd);
|
||||||
|
RootBeanDefinition resolvedInnerBd = bdvr.resolveInnerBean(null,innerBd, (name, mbd) -> {
|
||||||
|
assertThat(name).isNotNull().startsWith("(inner bean");
|
||||||
|
return mbd;
|
||||||
|
});
|
||||||
|
assertThat(resolvedInnerBd.getAttribute("test")).isEqualTo(42);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue