SPR-5899: Invoking methods or constructors passing null (including varargs support)
git-svn-id: https://src.springframework.org/svn/spring-framework/trunk@1478 50f2f4bb-b051-0410-bef5-90022cba6387
This commit is contained in:
parent
4f9c1bf118
commit
3150681307
|
|
@ -81,7 +81,7 @@ public class ConstructorReference extends SpelNodeImpl {
|
|||
TypedValue childValue = children[i + 1].getValueInternal(state);
|
||||
Object value = childValue.getValue();
|
||||
arguments[i] = value;
|
||||
argumentTypes[i] = (value==null?Object.class:value.getClass());
|
||||
argumentTypes[i] = (value==null?null:value.getClass());
|
||||
}
|
||||
|
||||
ConstructorExecutor executorToUse = this.cachedExecutor;
|
||||
|
|
|
|||
|
|
@ -92,7 +92,7 @@ public class MethodReference extends SpelNodeImpl {
|
|||
private Class<?>[] getTypes(Object... arguments) {
|
||||
Class<?>[] argumentTypes = new Class[arguments.length];
|
||||
for (int i = 0; i < arguments.length; i++) {
|
||||
argumentTypes[i] = (arguments[i]==null?Object.class:arguments[i].getClass());
|
||||
argumentTypes[i] = (arguments[i]==null?null:arguments[i].getClass());
|
||||
}
|
||||
return argumentTypes;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -57,19 +57,26 @@ public class ReflectionHelper {
|
|||
Class suppliedArg = suppliedArgTypes[i];
|
||||
Class expectedArg = expectedArgTypes[i];
|
||||
if (expectedArg != suppliedArg) {
|
||||
if (ClassUtils.isAssignable(expectedArg, suppliedArg)
|
||||
/* || isWidenableTo(expectedArg, suppliedArg) */) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, expectedArg)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
// The user may supply null - and that will be ok unless a primitive is expected
|
||||
if (suppliedArg==null) {
|
||||
if (expectedArg.isPrimitive()) {
|
||||
match=null;
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
} else {
|
||||
match = null;
|
||||
if (ClassUtils.isAssignable(expectedArg, suppliedArg)
|
||||
/* || isWidenableTo(expectedArg, suppliedArg) */) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, expectedArg)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
} else {
|
||||
match = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -113,23 +120,29 @@ public class ReflectionHelper {
|
|||
for (int i = 0; i < argCountUpToVarargs && match != null; i++) {
|
||||
Class suppliedArg = suppliedArgTypes[i];
|
||||
Class<?> expectedArg = expectedArgTypes[i];
|
||||
if (expectedArg != suppliedArg) {
|
||||
if (expectedArg.isAssignableFrom(suppliedArg) || ClassUtils.isAssignableValue(expectedArg, suppliedArg)) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
if (suppliedArg==null) {
|
||||
if (expectedArg.isPrimitive()) {
|
||||
match=null;
|
||||
}
|
||||
} else {
|
||||
if (expectedArg != suppliedArg) {
|
||||
if (expectedArg.isAssignableFrom(suppliedArg) || ClassUtils.isAssignableValue(expectedArg, suppliedArg)) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, expectedArg)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
} else {
|
||||
match = null;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, expectedArg)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
} else {
|
||||
match = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
// If already confirmed it cannot be a match, then returnW
|
||||
// If already confirmed it cannot be a match, then return
|
||||
if (match == null) {
|
||||
return null;
|
||||
}
|
||||
|
|
@ -147,19 +160,24 @@ public class ReflectionHelper {
|
|||
for (int i = expectedArgTypes.length - 1; i < suppliedArgTypes.length; i++) {
|
||||
Class suppliedArg = suppliedArgTypes[i];
|
||||
if (varargsParameterType != suppliedArg) {
|
||||
if (ClassUtils.isAssignable(varargsParameterType, suppliedArg)) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
if (suppliedArg==null) {
|
||||
if (varargsParameterType.isPrimitive()) {
|
||||
match=null;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, varargsParameterType)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
} else {
|
||||
if (ClassUtils.isAssignable(varargsParameterType, suppliedArg)) {
|
||||
if (match != ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
match = ArgsMatchKind.CLOSE;
|
||||
}
|
||||
} else if (typeConverter.canConvert(suppliedArg, varargsParameterType)) {
|
||||
if (argsRequiringConversion == null) {
|
||||
argsRequiringConversion = new ArrayList<Integer>();
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
} else {
|
||||
match = null;
|
||||
}
|
||||
argsRequiringConversion.add(i);
|
||||
match = ArgsMatchKind.REQUIRES_CONVERSION;
|
||||
}
|
||||
else {
|
||||
match = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -167,8 +185,7 @@ public class ReflectionHelper {
|
|||
|
||||
if (match == null) {
|
||||
return null;
|
||||
}
|
||||
else {
|
||||
} else {
|
||||
if (match == ArgsMatchKind.REQUIRES_CONVERSION) {
|
||||
int[] argsArray = new int[argsRequiringConversion.size()];
|
||||
for (int i = 0; i < argsRequiringConversion.size(); i++) {
|
||||
|
|
|
|||
|
|
@ -21,11 +21,14 @@ import junit.framework.Assert;
|
|||
import org.junit.Test;
|
||||
import org.springframework.expression.AccessException;
|
||||
import org.springframework.expression.EvaluationContext;
|
||||
import org.springframework.expression.EvaluationException;
|
||||
import org.springframework.expression.Expression;
|
||||
import org.springframework.expression.ParserContext;
|
||||
import org.springframework.expression.PropertyAccessor;
|
||||
import org.springframework.expression.spel.standard.SpelExpressionParser;
|
||||
import org.springframework.expression.spel.support.ReflectivePropertyResolver;
|
||||
import org.springframework.expression.spel.support.StandardEvaluationContext;
|
||||
import org.springframework.expression.spel.support.StandardTypeLocator;
|
||||
|
||||
/**
|
||||
* Tests based on Jiras up to the release of Spring 3.0.0
|
||||
|
|
@ -39,6 +42,148 @@ public class SpringEL300Tests extends ExpressionTestCase {
|
|||
evaluate("joinThreeStrings('a',null,'c')", "anullc", String.class);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testSPR5899() throws Exception {
|
||||
StandardEvaluationContext eContext = new StandardEvaluationContext(new Spr5899Class());
|
||||
Expression expr = new SpelExpressionParser().parse("tryToInvokeWithNull(12)");
|
||||
Assert.assertEquals(12,expr.getValue(eContext));
|
||||
expr = new SpelExpressionParser().parse("tryToInvokeWithNull(null)");
|
||||
Assert.assertEquals(null,expr.getValue(eContext));
|
||||
try {
|
||||
expr = new SpelExpressionParser().parse("tryToInvokeWithNull2(null)");
|
||||
expr.getValue();
|
||||
Assert.fail("Should have failed to find a method to which it could pass null");
|
||||
} catch (EvaluationException see) {
|
||||
// success
|
||||
}
|
||||
eContext.setTypeLocator(new MyTypeLocator());
|
||||
|
||||
// varargs
|
||||
expr = new SpelExpressionParser().parse("tryToInvokeWithNull3(null,'a','b')");
|
||||
Assert.assertEquals("ab",expr.getValue(eContext));
|
||||
|
||||
// varargs 2 - null is packed into the varargs
|
||||
expr = new SpelExpressionParser().parse("tryToInvokeWithNull3(12,'a',null,'c')");
|
||||
Assert.assertEquals("anullc",expr.getValue(eContext));
|
||||
|
||||
// check we can find the ctor ok
|
||||
expr = new SpelExpressionParser().parse("new Spr5899Class().toString()");
|
||||
Assert.assertEquals("instance",expr.getValue(eContext));
|
||||
|
||||
expr = new SpelExpressionParser().parse("new Spr5899Class(null).toString()");
|
||||
Assert.assertEquals("instance",expr.getValue(eContext));
|
||||
|
||||
// ctor varargs
|
||||
expr = new SpelExpressionParser().parse("new Spr5899Class(null,'a','b').toString()");
|
||||
Assert.assertEquals("instance",expr.getValue(eContext));
|
||||
|
||||
// ctor varargs 2
|
||||
expr = new SpelExpressionParser().parse("new Spr5899Class(null,'a', null, 'b').toString()");
|
||||
Assert.assertEquals("instance",expr.getValue(eContext));
|
||||
}
|
||||
|
||||
static class MyTypeLocator extends StandardTypeLocator {
|
||||
|
||||
public Class<?> findType(String typename) throws EvaluationException {
|
||||
if (typename.equals("Spr5899Class")) {
|
||||
return Spr5899Class.class;
|
||||
}
|
||||
return super.findType(typename);
|
||||
}
|
||||
}
|
||||
|
||||
static class Spr5899Class {
|
||||
public Spr5899Class() {}
|
||||
public Spr5899Class(Integer i) { }
|
||||
public Spr5899Class(Integer i, String... s) { }
|
||||
|
||||
public Integer tryToInvokeWithNull(Integer value) { return value; }
|
||||
public Integer tryToInvokeWithNull2(int i) { return new Integer(i); }
|
||||
public String tryToInvokeWithNull3(Integer value,String... strings) {
|
||||
StringBuilder sb = new StringBuilder();
|
||||
for (int i=0;i<strings.length;i++) {
|
||||
if (strings[i]==null) {
|
||||
sb.append("null");
|
||||
} else {
|
||||
sb.append(strings[i]);
|
||||
}
|
||||
}
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
public String toString() {
|
||||
return "instance";
|
||||
}
|
||||
}
|
||||
|
||||
// work in progress tests:
|
||||
// @Test
|
||||
// public void testSPR5804() throws ParseException, EvaluationException {
|
||||
// Map m = new HashMap();
|
||||
// m.put("foo",null);
|
||||
// StandardEvaluationContext eContext = new StandardEvaluationContext(m);
|
||||
// eContext.addPropertyAccessor(new MapAccessor());
|
||||
// Expression expr = new SpelExpressionParser().parse("foo");
|
||||
// Object o = expr.getValue(eContext);
|
||||
// }
|
||||
//
|
||||
//// jdbcProperties.username
|
||||
//
|
||||
// @Test
|
||||
// public void testSPR5847() throws ParseException, EvaluationException {
|
||||
// StandardEvaluationContext eContext = new StandardEvaluationContext(new TestProperties2());
|
||||
// Expression expr = new SpelExpressionParser().parse("jdbcProperties['username']");
|
||||
// eContext.addPropertyAccessor(new MapAccessor());
|
||||
// String name = expr.getValue(eContext,String.class);
|
||||
// Assert.assertEquals("Dave",name);
|
||||
//// System.out.println(o);
|
||||
//
|
||||
//
|
||||
//// Map m = new HashMap();
|
||||
//// m.put("jdbcProperties",new TestProperties());
|
||||
//// StandardEvaluationContext eContext = new StandardEvaluationContext(m);
|
||||
////// eContext.addPropertyAccessor(new MapAccessor());
|
||||
//// Expression expr = new SpelExpressionParser().parse("jdbcProperties.username");
|
||||
//// Object o = expr.getValue(eContext);
|
||||
//// System.out.println(o);
|
||||
// }
|
||||
//
|
||||
// static class TestProperties {
|
||||
// public String username = "Dave";
|
||||
// }
|
||||
//
|
||||
// static class TestProperties2 {
|
||||
// public Map jdbcProperties = new HashMap();
|
||||
// TestProperties2() {
|
||||
// jdbcProperties.put("username","Dave");
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// static class MapAccessor implements PropertyAccessor {
|
||||
//
|
||||
// public boolean canRead(EvaluationContext context, Object target, String name) throws AccessException {
|
||||
// return (((Map) target).containsKey(name));
|
||||
// }
|
||||
//
|
||||
// public TypedValue read(EvaluationContext context, Object target, String name) throws AccessException {
|
||||
// return new TypedValue(((Map) target).get(name),CommonTypeDescriptors.OBJECT_TYPE_DESCRIPTOR);
|
||||
// }
|
||||
//
|
||||
// public boolean canWrite(EvaluationContext context, Object target, String name) throws AccessException {
|
||||
// return true;
|
||||
// }
|
||||
//
|
||||
// @SuppressWarnings("unchecked")
|
||||
// public void write(EvaluationContext context, Object target, String name, Object newValue) throws AccessException {
|
||||
// ((Map) target).put(name, newValue);
|
||||
// }
|
||||
//
|
||||
// public Class[] getSpecificTargetClasses() {
|
||||
// return new Class[] {Map.class};
|
||||
// }
|
||||
//
|
||||
// }
|
||||
|
||||
@Test
|
||||
public void testNPE_SPR5673() throws Exception {
|
||||
ParserContext hashes = TemplateExpressionParsingTests.HASH_DELIMITED_PARSER_CONTEXT;
|
||||
|
|
|
|||
Loading…
Reference in New Issue