29
29
import org .codehaus .groovy .ast .stmt .Statement ;
30
30
import org .codehaus .groovy .ast .tools .ParameterUtils ;
31
31
import org .codehaus .groovy .control .CompilePhase ;
32
+ import org .codehaus .groovy .runtime .DefaultGroovyMethods ;
32
33
import org .codehaus .groovy .transform .ASTTransformation ;
33
34
import org .codehaus .groovy .transform .GroovyASTTransformation ;
34
35
import org .codehaus .groovy .vmplugin .VMPluginFactory ;
47
48
import java .util .Map ;
48
49
import java .util .Optional ;
49
50
import java .util .Set ;
51
+ import java .util .stream .IntStream ;
50
52
51
53
import static java .util .Arrays .stream ;
52
54
import static java .util .stream .Collectors .toList ;
@@ -1276,21 +1278,16 @@ public boolean hasPossibleMethod(final String name, final Expression arguments)
1276
1278
}
1277
1279
1278
1280
public MethodNode tryFindPossibleMethod (final String name , final Expression arguments ) {
1279
- if (!(arguments instanceof TupleExpression )) {
1280
- return null ;
1281
- }
1282
-
1283
- // TODO: this won't strictly be true when using list expansion in argument calls
1284
- TupleExpression args = (TupleExpression ) arguments ;
1285
- int nArgs = args .getExpressions ().size ();
1281
+ List <Expression > args = arguments instanceof TupleExpression ? ((TupleExpression ) arguments ).getExpressions () : Collections .singletonList (arguments );
1282
+ int nArgs = args .size (); // TODO: this isn't strictly accurate when using spread argument expansion
1286
1283
MethodNode method = null ;
1287
1284
1288
1285
for (ClassNode cn = this ; cn != null ; cn = cn .getSuperClass ()) {
1289
1286
for (MethodNode mn : cn .getDeclaredMethods (name )) {
1290
1287
if (hasCompatibleNumberOfArgs (mn , nArgs )) {
1291
1288
boolean match = true ;
1292
1289
for (int i = 0 ; i < nArgs ; i += 1 ) {
1293
- if (!hasCompatibleType (args , mn , i )) {
1290
+ if (!hasCompatibleType (args . get ( i ) , mn , i )) {
1294
1291
match = false ;
1295
1292
break ;
1296
1293
}
@@ -1314,6 +1311,18 @@ public MethodNode tryFindPossibleMethod(final String name, final Expression argu
1314
1311
}
1315
1312
}
1316
1313
1314
+ faces : if (method == null && DefaultGroovyMethods .asBoolean (getInterfaces ())) { // GROOVY-11323
1315
+ for (ClassNode cn : getAllInterfaces ()) {
1316
+ for (MethodNode mn : cn .getDeclaredMethods (name )) {
1317
+ if (mn .isPublic () && !mn .isStatic () && hasCompatibleNumberOfArgs (mn , nArgs ) && (nArgs == 0
1318
+ || IntStream .range (0 ,nArgs ).allMatch (i -> hasCompatibleType (args .get (i ),mn ,i )))) {
1319
+ method = mn ;
1320
+ break faces ;
1321
+ }
1322
+ }
1323
+ }
1324
+ }
1325
+
1317
1326
return method ;
1318
1327
}
1319
1328
@@ -1323,10 +1332,10 @@ private static boolean hasExactMatchingCompatibleType(final MethodNode match, fi
1323
1332
|| (i >= lastParamIndex && isPotentialVarArg (maybe , lastParamIndex ) && match .getParameters ()[i ].getType ().equals (maybe .getParameters ()[lastParamIndex ].getType ().getComponentType ()));
1324
1333
}
1325
1334
1326
- private static boolean hasCompatibleType (final TupleExpression args , final MethodNode method , final int i ) {
1335
+ private static boolean hasCompatibleType (final Expression arg , final MethodNode method , final int i ) {
1327
1336
int lastParamIndex = method .getParameters ().length - 1 ;
1328
- return (i <= lastParamIndex && args . getExpression ( i ) .getType ().isDerivedFrom (method .getParameters ()[i ].getType ()))
1329
- || (i >= lastParamIndex && isPotentialVarArg (method , lastParamIndex ) && args . getExpression ( i ) .getType ().isDerivedFrom (method .getParameters ()[lastParamIndex ].getType ().getComponentType ()));
1337
+ return (i <= lastParamIndex && arg .getType ().isDerivedFrom (method .getParameters ()[i ].getType ()))
1338
+ || (i >= lastParamIndex && isPotentialVarArg (method , lastParamIndex ) && arg .getType ().isDerivedFrom (method .getParameters ()[lastParamIndex ].getType ().getComponentType ()));
1330
1339
}
1331
1340
1332
1341
private static boolean hasCompatibleNumberOfArgs (final MethodNode method , final int nArgs ) {
0 commit comments