org.openrewrite.staticanalysis.UseLambdaForFunctionalInterface Maven / Gradle / Ivy
/*
* Copyright 2021 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.JavaIsoVisitor;
import org.openrewrite.java.JavaTemplate;
import org.openrewrite.java.JavaVisitor;
import org.openrewrite.java.RemoveUnusedImports;
import org.openrewrite.java.cleanup.UnnecessaryParenthesesVisitor;
import org.openrewrite.java.tree.*;
import org.openrewrite.marker.Markers;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Collectors;
import static java.util.Collections.singleton;
public class UseLambdaForFunctionalInterface extends Recipe {
@Override
public String getDisplayName() {
return "Use lambda expressions instead of anonymous classes";
}
@Override
public String getDescription() {
return "Instead of anonymous class declarations, use a lambda where possible. Using lambdas to replace " +
"anonymous classes can lead to more expressive and maintainable code, improve code readability, reduce " +
"code duplication, and achieve better performance in some cases.";
}
@Override
public Set getTags() {
return singleton("RSPEC-S1604");
}
@Override
public Duration getEstimatedEffortPerOccurrence() {
return Duration.ofMinutes(5);
}
@Override
public TreeVisitor, ExecutionContext> getVisitor() {
return Repeat.repeatUntilStable(new JavaVisitor() {
@Override
public J visitNewClass(J.NewClass newClass, ExecutionContext ctx) {
J.NewClass n = (J.NewClass) super.visitNewClass(newClass, ctx);
updateCursor(n);
if (n.getBody() != null &&
n.getBody().getStatements().size() == 1 &&
n.getBody().getStatements().get(0) instanceof J.MethodDeclaration &&
n.getClazz() != null) {
JavaType.FullyQualified type = TypeUtils.asFullyQualified(n.getClazz().getType());
if (type != null && type.getKind().equals(JavaType.Class.Kind.Interface)) {
JavaType.Method sam = getSamCompatible(type);
if (sam == null) {
return n;
}
if (usesThis(getCursor()) ||
shadowsLocalVariable(getCursor()) ||
usedAsStatement(getCursor()) ||
fieldInitializerReferencingUninitializedField(getCursor())) {
return n;
}
// The interface may be parameterized and that is needed to maintain type attribution:
JavaType.FullyQualified typedInterface = null;
JavaType.FullyQualified anonymousClass = TypeUtils.asFullyQualified(n.getType());
if (anonymousClass != null) {
typedInterface = anonymousClass.getInterfaces().stream().filter(i -> i.getFullyQualifiedName().equals(type.getFullyQualifiedName())).findFirst().orElse(null);
}
if (typedInterface == null) {
return n;
}
StringBuilder templateBuilder = new StringBuilder();
J.MethodDeclaration methodDeclaration = (J.MethodDeclaration) n.getBody().getStatements().get(0);
// If the functional interface method has type parameters, we can't replace it with a lambda.
if (methodDeclaration.getTypeParameters() != null && !methodDeclaration.getTypeParameters().isEmpty()) {
return n;
}
if (methodDeclaration.getParameters().get(0) instanceof J.Empty) {
templateBuilder.append("() -> {");
} else {
templateBuilder.append(methodDeclaration.getParameters().stream()
.map(param -> ((J.VariableDeclarations) param).getVariables().get(0).getSimpleName())
.collect(Collectors.joining(",", "(", ") -> {")));
}
JavaType returnType = sam.getReturnType();
if (!JavaType.Primitive.Void.equals(returnType)) {
templateBuilder.append("return ").append(valueOfType(returnType)).append(';');
}
templateBuilder.append('}');
J.Lambda lambda = JavaTemplate.builder(templateBuilder.toString())
.contextSensitive()
.build()
.apply(getCursor(), n.getCoordinates().replace());
lambda = lambda.withType(typedInterface);
lambda = (J.Lambda) new UnnecessaryParenthesesVisitor<>()
.visitNonNull(lambda, ctx, getCursor().getParentOrThrow());
J.Block lambdaBody = methodDeclaration.getBody();
assert lambdaBody != null;
lambda = lambda.withBody(lambdaBody.withPrefix(Space.format(" ")));
lambda = (J.Lambda) new LambdaBlockToExpression().getVisitor().visitNonNull(lambda, ctx, getCursor().getParentOrThrow());
doAfterVisit(new RemoveUnusedImports().getVisitor());
return autoFormat(maybeAddCast(lambda, newClass), ctx);
}
}
return n;
}
private J maybeAddCast(J.Lambda lambda, J.NewClass original) {
J parent = getCursor().getParentTreeCursor().getValue();
if (parent instanceof MethodCall) {
MethodCall method = (MethodCall) parent;
List arguments = method.getArguments();
for (int i = 0; i < arguments.size(); i++) {
Expression argument = arguments.get(i);
if (argument == original && methodArgumentRequiresCast(lambda, method, i) &&
original.getClazz() != null) {
return new J.TypeCast(
Tree.randomId(),
lambda.getPrefix(),
Markers.EMPTY,
new J.ControlParentheses<>(
Tree.randomId(),
Space.EMPTY,
Markers.EMPTY,
JRightPadded.build(original.getClazz())
),
lambda.withPrefix(Space.format(" "))
);
}
}
}
return lambda;
}
private boolean methodArgumentRequiresCast(J.Lambda lambda, MethodCall method, int argumentIndex) {
JavaType.FullyQualified lambdaType = TypeUtils.asFullyQualified(lambda.getType());
if (lambdaType == null) {
return false;
}
String lambdaFqn = lambdaType.getFullyQualifiedName();
JavaType.Method methodType = method.getMethodType();
if (methodType == null) {
return false;
}
if (!TypeUtils.isOfClassType(methodType.getParameterTypes().get(argumentIndex), lambdaFqn)) {
return true;
}
// look for ambiguous methods
int count = 0;
for (JavaType.Method maybeAmbiguous : methodType.getDeclaringType().getMethods()) {
if (methodType.getName().equals(maybeAmbiguous.getName()) &&
methodType.getParameterTypes().size() == maybeAmbiguous.getParameterTypes().size()) {
if (areMethodsAmbiguous(
getSamCompatible(methodType.getParameterTypes().get(argumentIndex)),
getSamCompatible(maybeAmbiguous.getParameterTypes().get(argumentIndex)))) {
count++;
}
}
}
if (count >= 2) {
return true;
}
return hasGenerics(lambda);
}
private boolean areMethodsAmbiguous(JavaType.@Nullable Method m1, JavaType.@Nullable Method m2) {
if (m1 == null || m2 == null) {
return false;
}
if (m1 == m2) {
return true;
}
for (int i = 0; i < m1.getParameterTypes().size(); i++) {
JavaType m1i = m1.getParameterTypes().get(i);
JavaType m2i = m2.getParameterTypes().get(i);
if (!TypeUtils.isAssignableTo(m1i, m2i) &&
!TypeUtils.isAssignableTo(m2i, m1i)) {
return false;
}
}
return true;
}
private String valueOfType(@Nullable JavaType type) {
JavaType.Primitive primitive = TypeUtils.asPrimitive(type);
if (primitive != null) {
switch (primitive) {
case Boolean:
return "true";
case Byte:
case Char:
case Int:
case Double:
case Float:
case Long:
case Short:
return "0";
case String:
case Null:
return "null";
case None:
case Void:
default:
return "";
}
}
return "null";
}
});
}
private static boolean usesThis(Cursor cursor) {
J.NewClass n = cursor.getValue();
assert n.getBody() != null;
AtomicBoolean hasThis = new AtomicBoolean(false);
new JavaVisitor() {
@Override
public J visitIdentifier(J.Identifier ident, Integer integer) {
if (ident.getSimpleName().equals("this")) {
hasThis.set(true);
}
return super.visitIdentifier(ident, integer);
}
}.visit(n.getBody(), 0, cursor);
return hasThis.get();
}
private static List parameterNames(J.MethodDeclaration method) {
return method.getParameters().stream()
.filter(J.VariableDeclarations.class::isInstance)
.map(v -> ((J.VariableDeclarations) v).getVariables().get(0).getSimpleName())
.collect(Collectors.toList());
}
// This does not recursive descend extended classes for inherited fields.
private static List classFields(J.ClassDeclaration classDeclaration) {
return classDeclaration.getBody().getStatements().stream()
.filter(J.VariableDeclarations.class::isInstance)
.map(v -> ((J.VariableDeclarations) v).getVariables().get(0).getSimpleName())
.collect(Collectors.toList());
}
private static boolean usedAsStatement(Cursor cursor) {
Iterator