org.openrewrite.staticanalysis.ReplaceLambdaWithMethodReference Maven / Gradle / Ivy
/*
* Copyright 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.openrewrite.staticanalysis;
import org.jspecify.annotations.Nullable;
import org.openrewrite.*;
import org.openrewrite.java.JavaTemplate;
import org.openrewrite.java.JavaVisitor;
import org.openrewrite.java.service.ImportService;
import org.openrewrite.java.tree.*;
import org.openrewrite.kotlin.KotlinVisitor;
import org.openrewrite.kotlin.tree.K;
import java.time.Duration;
import java.util.*;
import static org.openrewrite.staticanalysis.JavaElementFactory.*;
public class ReplaceLambdaWithMethodReference extends Recipe {
@Override
public String getDisplayName() {
return "Use method references in lambda";
}
@Override
public String getDescription() {
return "Replaces the single statement lambdas `o -> o instanceOf X`, `o -> (A) o`, `o -> System.out.println(o)`, `o -> o != null`, `o -> o == null` with the equivalent method reference.";
}
@Override
public Set getTags() {
return Collections.singleton("RSPEC-S1612");
}
@Override
public Duration getEstimatedEffortPerOccurrence() {
return Duration.ofMinutes(2);
}
@Override
public TreeVisitor, ExecutionContext> getVisitor() {
return new TreeVisitor() {
@Override
public @Nullable Tree visit(@Nullable Tree tree, ExecutionContext ctx, Cursor parent) {
if (tree instanceof J.CompilationUnit) {
return new ReplaceLambdaWithMethodReferenceJavaVisitor().visit(tree, ctx);
} else if (tree instanceof K.CompilationUnit) {
return new ReplaceLambdaWithMethodReferenceKotlinVisitor().visit(tree, ctx);
}
return tree;
}
};
}
private static class ReplaceLambdaWithMethodReferenceKotlinVisitor extends KotlinVisitor {
// Implement Me
}
private static class ReplaceLambdaWithMethodReferenceJavaVisitor extends JavaVisitor {
@Override
public J visitLambda(J.Lambda lambda, ExecutionContext ctx) {
J.Lambda l = (J.Lambda) super.visitLambda(lambda, ctx);
updateCursor(l);
J body = l.getBody();
if (body instanceof J.Block && ((J.Block) body).getStatements().size() == 1) {
Statement statement = ((J.Block) body).getStatements().get(0);
if (statement instanceof J.Return) {
body = ((J.Return) statement).getExpression();
} else {
body = statement;
}
}
if (body instanceof J.InstanceOf) {
J.InstanceOf instanceOf = (J.InstanceOf) body;
J j = instanceOf.getClazz();
if ((j instanceof J.Identifier || j instanceof J.FieldAccess) &&
instanceOf.getExpression() instanceof J.Identifier) {
J.FieldAccess classLiteral = newClassLiteral(((TypeTree) j).getType(), j instanceof J.FieldAccess);
if (classLiteral != null) {
//noinspection DataFlowIssue
JavaType.FullyQualified rawClassType = ((JavaType.Parameterized) classLiteral.getType()).getType();
Optional isInstanceMethod = rawClassType.getMethods().stream().filter(m -> m.getName().equals("isInstance")).findFirst();
if (isInstanceMethod.isPresent()) {
J.MemberReference updated = newInstanceMethodReference(classLiteral, isInstanceMethod.get(), lambda.getType()).withPrefix(lambda.getPrefix());
doAfterVisit(service(ImportService.class).shortenFullyQualifiedTypeReferencesIn(updated));
return updated;
}
}
}
return l;
} else if (body instanceof J.TypeCast && l.getParameters().getParameters().size() == 1) {
J.TypeCast cast = (J.TypeCast) body;
J param = l.getParameters().getParameters().get(0);
if (cast.getExpression() instanceof J.Identifier && param instanceof J.VariableDeclarations &&
((J.Identifier) cast.getExpression()).getSimpleName().equals(((J.VariableDeclarations) param).getVariables().get(0).getSimpleName())) {
J.ControlParentheses j = cast.getClazz();
J tree = j.getTree();
if ((tree instanceof J.Identifier || tree instanceof J.FieldAccess) &&
!(j.getType() instanceof JavaType.GenericTypeVariable)) {
J.FieldAccess classLiteral = newClassLiteral(((Expression) tree).getType(), tree instanceof J.FieldAccess);
if (classLiteral != null) {
//noinspection DataFlowIssue
JavaType.FullyQualified classType = ((JavaType.Parameterized) classLiteral.getType()).getType();
Optional castMethod = classType.getMethods().stream().filter(m -> m.getName().equals("cast")).findFirst();
if (castMethod.isPresent()) {
J.MemberReference updated = newInstanceMethodReference(classLiteral, castMethod.get(), lambda.getType()).withPrefix(lambda.getPrefix());
doAfterVisit(service(ImportService.class).shortenFullyQualifiedTypeReferencesIn(updated));
return updated;
}
}
}
}
return l;
}
String code = "";
if (body instanceof J.Binary) {
J.Binary binary = (J.Binary) body;
if ((binary.getOperator() == J.Binary.Type.Equal || binary.getOperator() == J.Binary.Type.NotEqual) &&
isNullCheck(binary.getLeft(), binary.getRight()) ||
isNullCheck(binary.getRight(), binary.getLeft())) {
code = J.Binary.Type.Equal == binary.getOperator() ? "java.util.Objects::isNull" :
"java.util.Objects::nonNull";
J updated = JavaTemplate.builder(code)
.contextSensitive()
.build()
.apply(getCursor(), l.getCoordinates().replace());
doAfterVisit(service(ImportService.class).shortenFullyQualifiedTypeReferencesIn(updated));
return updated;
}
} else if (body instanceof MethodCall) {
MethodCall method = (MethodCall) body;
if (method instanceof J.NewClass) {
J.NewClass nc = (J.NewClass) method;
if (nc.getBody() != null) {
return l;
} else {
if (isAMethodInvocationArgument(l, getCursor()) && nc.getType() instanceof JavaType.Class) {
JavaType.Class clazz = (JavaType.Class) nc.getType();
boolean hasMultipleConstructors = clazz.getMethods().stream().filter(JavaType.Method::isConstructor).count() > 1;
if (hasMultipleConstructors) {
return l;
}
}
}
} else if (method instanceof J.MemberReference) {
return l;
}
if (method.getMethodType() == null ||
hasSelectWithPotentialSideEffects(method) ||
hasSelectWhoseReferenceMightChange(method) ||
!methodArgumentsMatchLambdaParameters(method, lambda)) {
return l;
}
JavaType.Method methodType = method.getMethodType();
if (methodType != null && !isMethodReferenceAmbiguous(methodType)) {
Expression select =
method instanceof J.MethodInvocation ? ((J.MethodInvocation) method).getSelect() : null;
if (methodType.hasFlags(Flag.Static) ||
methodSelectMatchesFirstLambdaParameter(method, lambda)) {
if (method.getType() instanceof JavaType.Parameterized &&
((JavaType.Parameterized) method.getType()).getTypeParameters().stream()
.anyMatch(JavaType.GenericTypeVariable.class::isInstance)) {
return l;
}
J.MemberReference updated = newStaticMethodReference(methodType, true, lambda.getType()).withPrefix(lambda.getPrefix());
doAfterVisit(service(ImportService.class).shortenFullyQualifiedTypeReferencesIn(updated));
return updated;
} else if (method instanceof J.NewClass) {
NameTree clazz = ((J.NewClass) method).getClazz();
clazz = clazz instanceof J.ParameterizedType ? ((J.ParameterizedType) clazz).getClazz() : clazz;
return newInstanceMethodReference(clazz.withPrefix(Space.EMPTY), "new", methodType, lambda.getType()).withPrefix(lambda.getPrefix());
} else if (select != null) {
return newInstanceMethodReference(select, methodType, lambda.getType()).withPrefix(lambda.getPrefix());
} else {
Cursor owner = getCursor().dropParentUntil(is -> is instanceof J.ClassDeclaration ||
(is instanceof J.NewClass && ((J.NewClass) is).getBody() != null) ||
is instanceof J.Lambda);
return JavaElementFactory.newInstanceMethodReference(
JavaElementFactory.newThis(owner.getValue().getType()),
methodType,
lambda.getType()
).withPrefix(lambda.getPrefix());
}
}
}
return l;
}
private boolean hasSelectWithPotentialSideEffects(MethodCall method) {
return method instanceof J.MethodInvocation &&
((J.MethodInvocation) method).getSelect() instanceof MethodCall;
}
private boolean hasSelectWhoseReferenceMightChange(MethodCall method) {
if (method instanceof J.MethodInvocation) {
Expression select = ((J.MethodInvocation) method).getSelect();
if (select instanceof J.Identifier) {
JavaType.Variable fieldType = ((J.Identifier) select).getFieldType();
return fieldType != null && fieldType.getOwner() instanceof JavaType.Class && !fieldType.hasFlags(Flag.Final);
} else if (select instanceof J.FieldAccess) {
JavaType.Variable fieldType = ((J.FieldAccess) select).getName().getFieldType();
return fieldType != null && fieldType.getOwner() instanceof JavaType.Class && !fieldType.hasFlags(Flag.Final);
}
}
return false;
}
private boolean methodArgumentsMatchLambdaParameters(MethodCall method, J.Lambda lambda) {
JavaType.Method methodType = method.getMethodType();
if (methodType == null) {
return false;
}
boolean static_ = methodType.hasFlags(Flag.Static);
List methodArgs = getMethodArguments(method);
List lambdaParameters = getLambdaParameters(lambda);
if (methodArgs.isEmpty() && lambdaParameters.isEmpty()) {
return true;
}
if (!static_ && methodSelectMatchesFirstLambdaParameter(method, lambda)) {
methodArgs.add(0, ((J.MethodInvocation) method).getSelect());
}
if (methodArgs.size() != lambdaParameters.size()) {
return false;
}
for (int i = 0; i < lambdaParameters.size(); i++) {
JavaType lambdaParam = lambdaParameters.get(i).getVariableType();
if (!(methodArgs.get(i) instanceof J.Identifier)) {
return false;
}
JavaType methodArgument = ((J.Identifier) methodArgs.get(i)).getFieldType();
if (lambdaParam != methodArgument) {
return false;
}
}
return true;
}
private static List getMethodArguments(MethodCall method) {
List list = new ArrayList<>();
if (method instanceof J.MethodInvocation) {
// avoid additional `ArrayList` allocation by using `JContainer#getElements()`
for (Expression a : ((J.MethodInvocation) method).getPadding().getArguments().getElements()) {
if (!(a instanceof J.Empty)) {
list.add(a);
}
}
} else if (method instanceof J.NewClass) {
// avoid additional `ArrayList` allocation by using `JContainer#getElements()`
for (Expression a : ((J.NewClass) method).getPadding().getArguments().getElements()) {
if (!(a instanceof J.Empty)) {
list.add(a);
}
}
} else {
for (Expression a : method.getArguments()) {
if (!(a instanceof J.Empty)) {
list.add(a);
}
}
}
return list;
}
private static List getLambdaParameters(J.Lambda lambda) {
List list = new ArrayList<>();
for (J j : lambda.getParameters().getParameters()) {
if (j instanceof J.VariableDeclarations) {
J.VariableDeclarations.NamedVariable namedVariable = ((J.VariableDeclarations) j).getVariables().get(0);
list.add(namedVariable);
}
}
return list;
}
private boolean methodSelectMatchesFirstLambdaParameter(MethodCall method, J.Lambda lambda) {
if (!(method instanceof J.MethodInvocation) ||
!(((J.MethodInvocation) method).getSelect() instanceof J.Identifier) ||
lambda.getParameters().getParameters().isEmpty() ||
!(lambda.getParameters().getParameters().get(0) instanceof J.VariableDeclarations)) {
return false;
}
J.VariableDeclarations firstLambdaParameter = (J.VariableDeclarations) lambda.getParameters()
.getParameters().get(0);
return ((J.Identifier) ((J.MethodInvocation) method).getSelect()).getFieldType() ==
firstLambdaParameter.getVariables().get(0).getVariableType();
}
private boolean isNullCheck(J j1, J j2) {
return j1 instanceof J.Identifier && j2 instanceof J.Literal &&
"null".equals(((J.Literal) j2).getValueSource());
}
private boolean isMethodReferenceAmbiguous(JavaType.Method method) {
int count = 0;
for (JavaType.Method meth : method.getDeclaringType().getMethods()) {
if (meth.getName().equals(method.getName()) && !meth.getName().equals("println") && !meth.isConstructor()) {
if (++count > 1) {
return true;
}
}
}
return false;
}
}
private static boolean isAMethodInvocationArgument(J.Lambda lambda, Cursor cursor) {
Cursor parent = cursor.dropParentUntil(p -> p instanceof J.MethodInvocation || p instanceof J.CompilationUnit);
if (parent.getValue() instanceof J.MethodInvocation) {
J.MethodInvocation m = parent.getValue();
return m.getArguments().stream().anyMatch(arg -> arg == lambda);
}
return false;
}
}