Improve target detection for FactoryBeans with generic
Closes gh-28809
This commit is contained in:
parent
bbcc269487
commit
c1738aab0d
|
@ -72,23 +72,45 @@ class DefaultBeanRegistrationCodeFragments extends BeanRegistrationCodeFragments
|
||||||
public Class<?> getTarget(RegisteredBean registeredBean,
|
public Class<?> getTarget(RegisteredBean registeredBean,
|
||||||
Executable constructorOrFactoryMethod) {
|
Executable constructorOrFactoryMethod) {
|
||||||
|
|
||||||
Class<?> target = extractDeclaringClass(constructorOrFactoryMethod);
|
Class<?> target = extractDeclaringClass(registeredBean.getBeanType(),
|
||||||
|
constructorOrFactoryMethod);
|
||||||
while (target.getName().startsWith("java.") && registeredBean.isInnerBean()) {
|
while (target.getName().startsWith("java.") && registeredBean.isInnerBean()) {
|
||||||
target = registeredBean.getParent().getBeanClass();
|
target = registeredBean.getParent().getBeanClass();
|
||||||
}
|
}
|
||||||
return target;
|
return target;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Class<?> extractDeclaringClass(Executable executable) {
|
private Class<?> extractDeclaringClass(ResolvableType beanType, Executable executable) {
|
||||||
Class<?> declaringClass = ClassUtils.getUserClass(executable.getDeclaringClass());
|
Class<?> declaringClass = ClassUtils.getUserClass(executable.getDeclaringClass());
|
||||||
if (executable instanceof Constructor<?>
|
if (executable instanceof Constructor<?>
|
||||||
&& AccessVisibility.forMember(executable) == AccessVisibility.PUBLIC
|
&& AccessVisibility.forMember(executable) == AccessVisibility.PUBLIC
|
||||||
&& FactoryBean.class.isAssignableFrom(declaringClass)) {
|
&& FactoryBean.class.isAssignableFrom(declaringClass)) {
|
||||||
return ResolvableType.forType(declaringClass).as(FactoryBean.class).getGeneric(0).toClass();
|
return extractTargetClassFromFactoryBean(declaringClass, beanType);
|
||||||
}
|
}
|
||||||
return executable.getDeclaringClass();
|
return executable.getDeclaringClass();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract the target class of a public {@link FactoryBean} based on its
|
||||||
|
* constructor. If the implementation does not resolve the target class
|
||||||
|
* because it itself uses a generic, attempt to extract it from the
|
||||||
|
* bean type.
|
||||||
|
* @param factoryBeanType the factory bean type
|
||||||
|
* @param beanType the bean type
|
||||||
|
* @return the target class to use
|
||||||
|
*/
|
||||||
|
private Class<?> extractTargetClassFromFactoryBean(Class<?> factoryBeanType, ResolvableType beanType) {
|
||||||
|
ResolvableType target = ResolvableType.forType(factoryBeanType)
|
||||||
|
.as(FactoryBean.class).getGeneric(0);
|
||||||
|
if (target.getType().equals(Class.class)) {
|
||||||
|
return target.toClass();
|
||||||
|
}
|
||||||
|
else if (factoryBeanType.isAssignableFrom(beanType.toClass())) {
|
||||||
|
return beanType.as(FactoryBean.class).getGeneric(0).toClass();
|
||||||
|
}
|
||||||
|
return beanType.toClass();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public CodeBlock generateNewBeanDefinitionCode(GenerationContext generationContext,
|
public CodeBlock generateNewBeanDefinitionCode(GenerationContext generationContext,
|
||||||
ResolvableType beanType, BeanRegistrationCode beanRegistrationCode) {
|
ResolvableType beanType, BeanRegistrationCode beanRegistrationCode) {
|
||||||
|
|
|
@ -27,6 +27,7 @@ import org.springframework.beans.factory.support.RegisteredBean;
|
||||||
import org.springframework.beans.factory.support.RootBeanDefinition;
|
import org.springframework.beans.factory.support.RootBeanDefinition;
|
||||||
import org.springframework.beans.testfixture.beans.factory.DummyFactory;
|
import org.springframework.beans.testfixture.beans.factory.DummyFactory;
|
||||||
import org.springframework.beans.testfixture.beans.factory.aot.MockBeanRegistrationsCode;
|
import org.springframework.beans.testfixture.beans.factory.aot.MockBeanRegistrationsCode;
|
||||||
|
import org.springframework.core.ResolvableType;
|
||||||
import org.springframework.core.testfixture.aot.generate.TestGenerationContext;
|
import org.springframework.core.testfixture.aot.generate.TestGenerationContext;
|
||||||
import org.springframework.util.ReflectionUtils;
|
import org.springframework.util.ReflectionUtils;
|
||||||
|
|
||||||
|
@ -57,6 +58,29 @@ class DefaultBeanRegistrationCodeFragmentsTests {
|
||||||
SimpleBeanFactoryBean.class.getDeclaredConstructors()[0])).isEqualTo(SimpleBean.class);
|
SimpleBeanFactoryBean.class.getDeclaredConstructors()[0])).isEqualTo(SimpleBean.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getTargetOnConstructorToPublicGenericFactoryBeanExtractTargetFromFactoryBeanType() {
|
||||||
|
RegisteredBean registeredBean = registerTestBean(ResolvableType
|
||||||
|
.forClassWithGenerics(GenericFactoryBean.class, SimpleBean.class));
|
||||||
|
assertThat(createInstance(registeredBean).getTarget(registeredBean,
|
||||||
|
GenericFactoryBean.class.getDeclaredConstructors()[0])).isEqualTo(SimpleBean.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getTargetOnConstructorToPublicGenericFactoryBeanWithBoundExtractTargetFromFactoryBeanType() {
|
||||||
|
RegisteredBean registeredBean = registerTestBean(ResolvableType
|
||||||
|
.forClassWithGenerics(NumberFactoryBean.class, Integer.class));
|
||||||
|
assertThat(createInstance(registeredBean).getTarget(registeredBean,
|
||||||
|
NumberFactoryBean.class.getDeclaredConstructors()[0])).isEqualTo(Integer.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getTargetOnConstructorToPublicGenericFactoryBeanUseBeanTypeAsFallback() {
|
||||||
|
RegisteredBean registeredBean = registerTestBean(SimpleBean.class);
|
||||||
|
assertThat(createInstance(registeredBean).getTarget(registeredBean,
|
||||||
|
GenericFactoryBean.class.getDeclaredConstructors()[0])).isEqualTo(SimpleBean.class);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getTargetOnConstructorToProtectedFactoryBean() {
|
void getTargetOnConstructorToProtectedFactoryBean() {
|
||||||
RegisteredBean registeredBean = registerTestBean(SimpleBean.class);
|
RegisteredBean registeredBean = registerTestBean(SimpleBean.class);
|
||||||
|
@ -138,6 +162,12 @@ class DefaultBeanRegistrationCodeFragmentsTests {
|
||||||
return RegisteredBean.of(this.beanFactory, "testBean");
|
return RegisteredBean.of(this.beanFactory, "testBean");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private RegisteredBean registerTestBean(ResolvableType beanType) {
|
||||||
|
this.beanFactory.registerBeanDefinition("testBean",
|
||||||
|
new RootBeanDefinition(beanType));
|
||||||
|
return RegisteredBean.of(this.beanFactory, "testBean");
|
||||||
|
}
|
||||||
|
|
||||||
private BeanRegistrationCodeFragments createInstance(RegisteredBean registeredBean) {
|
private BeanRegistrationCodeFragments createInstance(RegisteredBean registeredBean) {
|
||||||
return new DefaultBeanRegistrationCodeFragments(this.beanRegistrationsCode, registeredBean,
|
return new DefaultBeanRegistrationCodeFragments(this.beanRegistrationsCode, registeredBean,
|
||||||
new BeanDefinitionMethodGeneratorFactory(this.beanFactory));
|
new BeanDefinitionMethodGeneratorFactory(this.beanFactory));
|
||||||
|
|
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
* 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.aot;
|
||||||
|
|
||||||
|
import org.springframework.beans.BeanUtils;
|
||||||
|
import org.springframework.beans.factory.FactoryBean;
|
||||||
|
import org.springframework.lang.Nullable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A public {@link FactoryBean} with a generic type.
|
||||||
|
*
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
*/
|
||||||
|
public class GenericFactoryBean<T> implements FactoryBean<T> {
|
||||||
|
|
||||||
|
private final Class<T> beanType;
|
||||||
|
|
||||||
|
public GenericFactoryBean(Class<T> beanType) {
|
||||||
|
this.beanType = beanType;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public T getObject() throws Exception {
|
||||||
|
return BeanUtils.instantiateClass(this.beanType);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public Class<?> getObjectType() {
|
||||||
|
return this.beanType;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
/*
|
||||||
|
* 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.aot;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link GenericFactoryBean} that has a bound for the target type.
|
||||||
|
*
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
*/
|
||||||
|
public class NumberFactoryBean<T extends Number> extends GenericFactoryBean<T> {
|
||||||
|
|
||||||
|
public NumberFactoryBean(Class<T> beanType) {
|
||||||
|
super(beanType);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -19,20 +19,14 @@ package org.springframework.beans.factory.aot;
|
||||||
import org.springframework.beans.factory.FactoryBean;
|
import org.springframework.beans.factory.FactoryBean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A public {@link FactoryBean}.
|
* A public {@link FactoryBean} with a resolved generic for {@link GenericFactoryBean}.
|
||||||
*
|
*
|
||||||
* @author Stephane Nicoll
|
* @author Stephane Nicoll
|
||||||
*/
|
*/
|
||||||
public class SimpleBeanFactoryBean implements FactoryBean<SimpleBean> {
|
public class SimpleBeanFactoryBean extends GenericFactoryBean<SimpleBean> {
|
||||||
|
|
||||||
@Override
|
public SimpleBeanFactoryBean() {
|
||||||
public SimpleBean getObject() throws Exception {
|
super(SimpleBean.class);
|
||||||
return new SimpleBean();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Class<?> getObjectType() {
|
|
||||||
return SimpleBean.class;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue