instantiateUsingFactoryMethod avoids NPE and reports argument types in case of explicitArgs and resolved generic arguments as well

Issue: SPR-11517
This commit is contained in:
Juergen Hoeller 2014-03-05 16:16:29 +01:00
parent 9891bdc7b4
commit ad317774fb
3 changed files with 55 additions and 15 deletions

View File

@ -523,24 +523,30 @@ class ConstructorResolver {
} }
if (factoryMethodToUse == null) { if (factoryMethodToUse == null) {
boolean hasArgs = (resolvedValues.getArgumentCount() > 0); List<String> argTypes = new ArrayList<String>(minNrOfArgs);
String argDesc = ""; if (explicitArgs != null) {
if (hasArgs) { for (Object arg : explicitArgs) {
List<String> argTypes = new ArrayList<String>(); argTypes.add(arg != null ? arg.getClass().getSimpleName() : "null");
for (ValueHolder value : resolvedValues.getIndexedArgumentValues().values()) { }
String argType = (value.getType() != null ? }
ClassUtils.getShortName(value.getType()) : value.getValue().getClass().getSimpleName()); else {
Set<ValueHolder> valueHolders = new LinkedHashSet<ValueHolder>(resolvedValues.getArgumentCount());
valueHolders.addAll(resolvedValues.getIndexedArgumentValues().values());
valueHolders.addAll(resolvedValues.getGenericArgumentValues());
for (ValueHolder value : valueHolders) {
String argType = (value.getType() != null ? ClassUtils.getShortName(value.getType()) :
(value.getValue() != null ? value.getValue().getClass().getSimpleName() : "null"));
argTypes.add(argType); argTypes.add(argType);
} }
argDesc = StringUtils.collectionToCommaDelimitedString(argTypes);
} }
String argDesc = StringUtils.collectionToCommaDelimitedString(argTypes);
throw new BeanCreationException(mbd.getResourceDescription(), beanName, throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"No matching factory method found: " + "No matching factory method found: " +
(mbd.getFactoryBeanName() != null ? (mbd.getFactoryBeanName() != null ?
"factory bean '" + mbd.getFactoryBeanName() + "'; " : "") + "factory bean '" + mbd.getFactoryBeanName() + "'; " : "") +
"factory method '" + mbd.getFactoryMethodName() + "(" + argDesc + ")'. " + "factory method '" + mbd.getFactoryMethodName() + "(" + argDesc + ")'. " +
"Check that a method with the specified name " + "Check that a method with the specified name " +
(hasArgs ? "and arguments " : "") + (minNrOfArgs > 0 ? "and arguments " : "") +
"exists and that it is " + "exists and that it is " +
(isStatic ? "static" : "non-static") + "."); (isStatic ? "static" : "non-static") + ".");
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2013 the original author or authors. * Copyright 2002-2014 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -21,7 +21,6 @@ import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Properties; import java.util.Properties;
import static org.junit.Assert.*;
import org.junit.Test; import org.junit.Test;
import org.springframework.beans.factory.BeanCreationException; import org.springframework.beans.factory.BeanCreationException;
@ -30,6 +29,8 @@ import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.core.io.ClassPathResource; import org.springframework.core.io.ClassPathResource;
import org.springframework.tests.sample.beans.TestBean; import org.springframework.tests.sample.beans.TestBean;
import static org.junit.Assert.*;
/** /**
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Chris Beams * @author Chris Beams
@ -259,6 +260,34 @@ public class FactoryMethodTests {
} }
} }
@Test
public void testNonExistingFactoryMethod() {
DefaultListableBeanFactory xbf = new DefaultListableBeanFactory();
XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(xbf);
reader.loadBeanDefinitions(new ClassPathResource("factory-methods.xml", getClass()));
try {
xbf.getBean("invalidPrototype");
fail("Should have thrown BeanCreationException");
}
catch (BeanCreationException ex) {
assertTrue(ex.getMessage().contains("nonExisting(TestBean)"));
}
}
@Test
public void testFactoryMethodArgumentsForNonExistingMethod() {
DefaultListableBeanFactory xbf = new DefaultListableBeanFactory();
XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(xbf);
reader.loadBeanDefinitions(new ClassPathResource("factory-methods.xml", getClass()));
try {
xbf.getBean("invalidPrototype", new TestBean());
fail("Should have thrown BeanCreationException");
}
catch (BeanCreationException ex) {
assertTrue(ex.getMessage().contains("nonExisting(TestBean)"));
}
}
@Test @Test
public void testCanSpecifyFactoryMethodArgumentsOnFactoryMethodPrototype() { public void testCanSpecifyFactoryMethodArgumentsOnFactoryMethodPrototype() {
DefaultListableBeanFactory xbf = new DefaultListableBeanFactory(); DefaultListableBeanFactory xbf = new DefaultListableBeanFactory();
@ -360,7 +389,9 @@ public class FactoryMethodTests {
} }
class MailSession { class MailSession {
private Properties props; private Properties props;
private MailSession() { private MailSession() {
@ -377,6 +408,6 @@ class MailSession {
} }
public Object getProperty(String key) { public Object getProperty(String key) {
return props.get(key); return this.props.get(key);
} }
} }

View File

@ -84,6 +84,11 @@
<property name="stringValue"><value>testBeanOnlyPrototypeDISetterString</value></property> <property name="stringValue"><value>testBeanOnlyPrototypeDISetterString</value></property>
</bean> </bean>
<bean id="invalidPrototype" class="org.springframework.beans.factory.xml.FactoryMethods"
factory-method="nonExisting" scope="prototype">
<constructor-arg><ref local="juergen"/></constructor-arg>
</bean>
<bean id="fullPrototype" class="org.springframework.beans.factory.xml.FactoryMethods" <bean id="fullPrototype" class="org.springframework.beans.factory.xml.FactoryMethods"
factory-method="newInstance" scope="prototype"> factory-method="newInstance" scope="prototype">
<constructor-arg type="int"><value>27</value></constructor-arg> <constructor-arg type="int"><value>27</value></constructor-arg>
@ -120,9 +125,7 @@
<constructor-arg><value type="java.lang.Integer">33</value></constructor-arg> <constructor-arg><value type="java.lang.Integer">33</value></constructor-arg>
</bean> </bean>
<bean id="instanceFactoryMethodWithoutArgs" <bean id="instanceFactoryMethodWithoutArgs" factory-bean="instanceFactory" factory-method="defaultInstance"/>
factory-bean="instanceFactory"
factory-method="defaultInstance"/>
<!-- Unnamed bean with factory-bean declaration --> <!-- Unnamed bean with factory-bean declaration -->
<bean factory-bean="instanceFactory" factory-method="defaultInstance"/> <bean factory-bean="instanceFactory" factory-method="defaultInstance"/>