Skip to content

Commit e03e62c

Browse files
committed
Test status quo for null in varargs in SpEL expressions
This commit also points out that `null` supplied as a single value for a varargs array of type Optional will be kept as `null` instead of being converted to Optional.empty(); whereas, if more than one value is passed to such a varargs array a null value will be properly converted to Optional.empty(). See gh-27719
1 parent 96a7fc6 commit e03e62c

File tree

4 files changed

+85
-58
lines changed

4 files changed

+85
-58
lines changed

spring-expression/src/test/java/org/springframework/expression/spel/MethodInvocationTests.java

Lines changed: 46 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2019 the original author or authors.
2+
* Copyright 2002-2021 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.
@@ -46,6 +46,7 @@
4646
*
4747
* @author Andy Clement
4848
* @author Phillip Webb
49+
* @author Sam Brannen
4950
*/
5051
public class MethodInvocationTests extends AbstractExpressionTests {
5152

@@ -232,26 +233,54 @@ public void testAddingMethodResolvers() {
232233

233234
@Test
234235
public void testVarargsInvocation01() {
235-
// Calling 'public int aVarargsMethod(String... strings)'
236-
//evaluate("aVarargsMethod('a','b','c')", 3, Integer.class);
237-
//evaluate("aVarargsMethod('a')", 1, Integer.class);
238-
evaluate("aVarargsMethod()", 0, Integer.class);
239-
evaluate("aVarargsMethod(1,2,3)", 3, Integer.class); // all need converting to strings
240-
evaluate("aVarargsMethod(1)", 1, Integer.class); // needs string conversion
241-
evaluate("aVarargsMethod(1,'a',3.0d)", 3, Integer.class); // first and last need conversion
242-
// evaluate("aVarargsMethod(new String[]{'a','b','c'})", 3, Integer.class);
236+
// Calling 'public String aVarargsMethod(String... strings)'
237+
evaluate("aVarargsMethod('a','b','c')", "[a, b, c]", String.class);
238+
evaluate("aVarargsMethod('a')", "[a]", String.class);
239+
evaluate("aVarargsMethod()", "[]", String.class);
240+
evaluate("aVarargsMethod(1,2,3)", "[1, 2, 3]", String.class); // all need converting to strings
241+
evaluate("aVarargsMethod(1)", "[1]", String.class); // needs string conversion
242+
evaluate("aVarargsMethod(1,'a',3.0d)", "[1, a, 3.0]", String.class); // first and last need conversion
243+
evaluate("aVarargsMethod(new String[]{'a','b','c'})", "[a, b, c]", String.class);
244+
evaluate("aVarargsMethod(new String[]{})", "[]", String.class);
245+
evaluate("aVarargsMethod(null)", "[null]", String.class);
246+
evaluate("aVarargsMethod(null,'a')", "[null, a]", String.class);
247+
evaluate("aVarargsMethod('a',null,'b')", "[a, null, b]", String.class);
243248
}
244249

245250
@Test
246251
public void testVarargsInvocation02() {
247-
// Calling 'public int aVarargsMethod2(int i, String... strings)' - returns int+length_of_strings
248-
evaluate("aVarargsMethod2(5,'a','b','c')", 8, Integer.class);
249-
evaluate("aVarargsMethod2(2,'a')", 3, Integer.class);
250-
evaluate("aVarargsMethod2(4)", 4, Integer.class);
251-
evaluate("aVarargsMethod2(8,2,3)", 10, Integer.class);
252-
evaluate("aVarargsMethod2(9)", 9, Integer.class);
253-
evaluate("aVarargsMethod2(2,'a',3.0d)", 4, Integer.class);
254-
// evaluate("aVarargsMethod2(8,new String[]{'a','b','c'})", 11, Integer.class);
252+
// Calling 'public String aVarargsMethod2(int i, String... strings)'
253+
evaluate("aVarargsMethod2(5,'a','b','c')", "5-[a, b, c]", String.class);
254+
evaluate("aVarargsMethod2(2,'a')", "2-[a]", String.class);
255+
evaluate("aVarargsMethod2(4)", "4-[]", String.class);
256+
evaluate("aVarargsMethod2(8,2,3)", "8-[2, 3]", String.class);
257+
evaluate("aVarargsMethod2(2,'a',3.0d)", "2-[a, 3.0]", String.class);
258+
evaluate("aVarargsMethod2(8,new String[]{'a','b','c'})", "8-[a, b, c]", String.class);
259+
evaluate("aVarargsMethod2(8,new String[]{})", "8-[]", String.class);
260+
evaluate("aVarargsMethod2(8,null)", "8-[null]", String.class);
261+
evaluate("aVarargsMethod2(8,null,'a')", "8-[null, a]", String.class);
262+
evaluate("aVarargsMethod2(8,'a',null,'b')", "8-[a, null, b]", String.class);
263+
}
264+
265+
@Test
266+
public void testVarargsOptionalInvocation() {
267+
// Calling 'public String optionalVarargsMethod(Optional<String>... values)'
268+
evaluate("optionalVarargsMethod()", "[]", String.class);
269+
evaluate("optionalVarargsMethod(new String[0])", "[]", String.class);
270+
evaluate("optionalVarargsMethod('a')", "[Optional[a]]", String.class);
271+
evaluate("optionalVarargsMethod('a','b','c')", "[Optional[a], Optional[b], Optional[c]]", String.class);
272+
evaluate("optionalVarargsMethod(9)", "[Optional[9]]", String.class);
273+
evaluate("optionalVarargsMethod(2,3)", "[Optional[2], Optional[3]]", String.class);
274+
evaluate("optionalVarargsMethod('a',3.0d)", "[Optional[a], Optional[3.0]]", String.class);
275+
evaluate("optionalVarargsMethod(new String[]{'a','b','c'})", "[Optional[a], Optional[b], Optional[c]]", String.class);
276+
// The following should actually evaluate to [Optional.empty] instead of [null],
277+
// but ReflectionHelper.convertArguments() passes the array type instead of
278+
// the array's component type as the target type to the ConversionService, and
279+
// GenericConversionService.convertNullSource() therefore fails to convert null
280+
// to Optional.empty().
281+
evaluate("optionalVarargsMethod(null)", "[null]", String.class);
282+
evaluate("optionalVarargsMethod(null,'a')", "[Optional.empty, Optional[a]]", String.class);
283+
evaluate("optionalVarargsMethod('a',null,'b')", "[Optional[a], Optional.empty, Optional[b]]", String.class);
255284
}
256285

257286
@Test

spring-expression/src/test/java/org/springframework/expression/spel/TestScenarioCreator.java

Lines changed: 10 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2015 the original author or authors.
2+
* Copyright 2002-2021 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.
@@ -16,6 +16,7 @@
1616

1717
package org.springframework.expression.spel;
1818

19+
import java.util.Arrays;
1920
import java.util.GregorianCalendar;
2021

2122
import org.springframework.expression.spel.support.StandardEvaluationContext;
@@ -51,10 +52,10 @@ private static void populateFunctions(StandardEvaluationContext testContext) {
5152
TestScenarioCreator.class.getDeclaredMethod("reverseInt", Integer.TYPE, Integer.TYPE, Integer.TYPE));
5253
testContext.registerFunction("reverseString",
5354
TestScenarioCreator.class.getDeclaredMethod("reverseString", String.class));
54-
testContext.registerFunction("varargsFunctionReverseStringsAndMerge",
55-
TestScenarioCreator.class.getDeclaredMethod("varargsFunctionReverseStringsAndMerge", String[].class));
56-
testContext.registerFunction("varargsFunctionReverseStringsAndMerge2",
57-
TestScenarioCreator.class.getDeclaredMethod("varargsFunctionReverseStringsAndMerge2", Integer.TYPE, String[].class));
55+
testContext.registerFunction("varargsFunction",
56+
TestScenarioCreator.class.getDeclaredMethod("varargsFunction", String[].class));
57+
testContext.registerFunction("varargsFunction2",
58+
TestScenarioCreator.class.getDeclaredMethod("varargsFunction2", Integer.TYPE, String[].class));
5859
}
5960
catch (Exception ex) {
6061
throw new IllegalStateException(ex);
@@ -108,25 +109,12 @@ public static String reverseString(String input) {
108109
return backwards.toString();
109110
}
110111

111-
public static String varargsFunctionReverseStringsAndMerge(String... strings) {
112-
StringBuilder sb = new StringBuilder();
113-
if (strings != null) {
114-
for (int i = strings.length - 1; i >= 0; i--) {
115-
sb.append(strings[i]);
116-
}
117-
}
118-
return sb.toString();
112+
public static String varargsFunction(String... strings) {
113+
return Arrays.toString(strings);
119114
}
120115

121-
public static String varargsFunctionReverseStringsAndMerge2(int j, String... strings) {
122-
StringBuilder sb = new StringBuilder();
123-
sb.append(j);
124-
if (strings != null) {
125-
for (int i = strings.length - 1; i >= 0; i--) {
126-
sb.append(strings[i]);
127-
}
128-
}
129-
return sb.toString();
116+
public static String varargsFunction2(int i, String... strings) {
117+
return String.valueOf(i) + "-" + Arrays.toString(strings);
130118
}
131119

132120
}

spring-expression/src/test/java/org/springframework/expression/spel/VariableAndFunctionTests.java

Lines changed: 17 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -58,16 +58,23 @@ public void testFunctionAccess02() {
5858

5959
@Test
6060
public void testCallVarargsFunction() {
61-
evaluate("#varargsFunctionReverseStringsAndMerge('a','b','c')", "cba", String.class);
62-
evaluate("#varargsFunctionReverseStringsAndMerge('a')", "a", String.class);
63-
evaluate("#varargsFunctionReverseStringsAndMerge()", "", String.class);
64-
evaluate("#varargsFunctionReverseStringsAndMerge('b',25)", "25b", String.class);
65-
evaluate("#varargsFunctionReverseStringsAndMerge(25)", "25", String.class);
66-
evaluate("#varargsFunctionReverseStringsAndMerge2(1,'a','b','c')", "1cba", String.class);
67-
evaluate("#varargsFunctionReverseStringsAndMerge2(2,'a')", "2a", String.class);
68-
evaluate("#varargsFunctionReverseStringsAndMerge2(3)", "3", String.class);
69-
evaluate("#varargsFunctionReverseStringsAndMerge2(4,'b',25)", "425b", String.class);
70-
evaluate("#varargsFunctionReverseStringsAndMerge2(5,25)", "525", String.class);
61+
evaluate("#varargsFunction()", "[]", String.class);
62+
evaluate("#varargsFunction(new String[0])", "[]", String.class);
63+
evaluate("#varargsFunction('a','b','c')", "[a, b, c]", String.class);
64+
evaluate("#varargsFunction('a')", "[a]", String.class);
65+
evaluate("#varargsFunction('b',25)", "[b, 25]", String.class);
66+
evaluate("#varargsFunction(25)", "[25]", String.class);
67+
evaluate("#varargsFunction(null)", "[null]", String.class);
68+
evaluate("#varargsFunction('a',null,'b')", "[a, null, b]", String.class);
69+
70+
evaluate("#varargsFunction2(9)", "9-[]", String.class);
71+
evaluate("#varargsFunction2(9, new String[0])", "9-[]", String.class);
72+
evaluate("#varargsFunction2(9,'a','b','c')", "9-[a, b, c]", String.class);
73+
evaluate("#varargsFunction2(9,'a')", "9-[a]", String.class);
74+
evaluate("#varargsFunction2(9,'b',25)", "9-[b, 25]", String.class);
75+
evaluate("#varargsFunction2(9,25)", "9-[25]", String.class);
76+
evaluate("#varargsFunction2(9,null)", "9-[null]", String.class);
77+
evaluate("#varargsFunction2(9,'a',null,'b')", "9-[a, null, b]", String.class);
7178
}
7279

7380
@Test

spring-expression/src/test/java/org/springframework/expression/spel/testresources/Inventor.java

Lines changed: 12 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2016 the original author or authors.
2+
* Copyright 2002-2021 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.
@@ -17,11 +17,13 @@
1717
package org.springframework.expression.spel.testresources;
1818

1919
import java.util.ArrayList;
20+
import java.util.Arrays;
2021
import java.util.Date;
2122
import java.util.HashMap;
2223
import java.util.LinkedHashMap;
2324
import java.util.List;
2425
import java.util.Map;
26+
import java.util.Optional;
2527

2628
import org.springframework.util.ObjectUtils;
2729

@@ -190,16 +192,17 @@ public String joinThreeStrings(String a, String b, String c) {
190192
return a + b + c;
191193
}
192194

193-
public int aVarargsMethod(String... strings) {
194-
if (strings == null)
195-
return 0;
196-
return strings.length;
195+
public String aVarargsMethod(String... strings) {
196+
return Arrays.toString(strings);
197197
}
198198

199-
public int aVarargsMethod2(int i, String... strings) {
200-
if (strings == null)
201-
return i;
202-
return strings.length + i;
199+
public String aVarargsMethod2(int i, String... strings) {
200+
return String.valueOf(i) + "-" + Arrays.toString(strings);
201+
}
202+
203+
@SuppressWarnings("unchecked")
204+
public String optionalVarargsMethod(Optional<String>... values) {
205+
return Arrays.toString(values);
203206
}
204207

205208
public Inventor(String... strings) {

0 commit comments

Comments
 (0)