org.openrewrite.staticanalysis.RemoveUnusedPrivateFields 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 lombok.EqualsAndHashCode;
import lombok.Value;
import org.jspecify.annotations.Nullable;
import org.openrewrite.Cursor;
import org.openrewrite.ExecutionContext;
import org.openrewrite.Recipe;
import org.openrewrite.TreeVisitor;
import org.openrewrite.java.AnnotationMatcher;
import org.openrewrite.java.JavaIsoVisitor;
import org.openrewrite.java.JavaVisitor;
import org.openrewrite.java.service.AnnotationService;
import org.openrewrite.java.tree.J;
import org.openrewrite.java.tree.Space;
import org.openrewrite.java.tree.Statement;
import org.openrewrite.java.tree.TypeUtils;
import java.time.Duration;
import java.util.*;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Collectors;
@Value
@EqualsAndHashCode(callSuper = false)
public class RemoveUnusedPrivateFields extends Recipe {
private static final AnnotationMatcher LOMBOK_ANNOTATION = new AnnotationMatcher("@lombok.*");
@Override
public String getDisplayName() {
return "Remove unused private fields";
}
@Override
public String getDescription() {
return "If a private field is declared but not used in the program, it can be considered dead code and should therefore be removed.";
}
@Override
public Set getTags() {
return Collections.singleton("RSPEC-S1068");
}
@Override
public Duration getEstimatedEffortPerOccurrence() {
return Duration.ofMinutes(5);
}
@Override
public TreeVisitor, ExecutionContext> getVisitor() {
return new JavaIsoVisitor() {
@Value
class CheckField {
J.VariableDeclarations declarations;
@Nullable Statement nextStatement;
}
@Override
public J.ClassDeclaration visitClassDeclaration(J.ClassDeclaration classDecl, ExecutionContext ctx) {
J.ClassDeclaration cd = super.visitClassDeclaration(classDecl, ctx);
// Do not remove fields if class has Lombok @Data annotation
Iterator clz = getCursor().getPathAsCursors(c -> c.getValue() instanceof J.ClassDeclaration);
if (clz.hasNext() && service(AnnotationService.class).matches(clz.next(), LOMBOK_ANNOTATION)) {
return cd;
}
List checkFields = new ArrayList<>();
// Do not remove fields with `serialVersionUID` name.
boolean skipSerialVersionUID = cd.getType() == null ||
cd.getType().isAssignableTo("java.io.Serializable");
List statements = cd.getBody().getStatements();
for (int i = 0; i < statements.size(); i++) {
Statement statement = statements.get(i);
if (statement instanceof J.VariableDeclarations) {
J.VariableDeclarations vd = (J.VariableDeclarations) statement;
// RSPEC-S1068 does not apply serialVersionUID of Serializable classes, or fields with annotations.
if (!(skipSerialVersionUID && isSerialVersionUid(vd)) &&
vd.getLeadingAnnotations().isEmpty() &&
vd.hasModifier(J.Modifier.Type.Private)) {
Statement nextStatement = i < statements.size() - 1 ? statements.get(i + 1) : null;
checkFields.add(new CheckField(vd, nextStatement));
}
} else if (statement instanceof J.MethodDeclaration) {
// RSPEC-S1068 does not apply fields from classes with native methods.
J.MethodDeclaration md = (J.MethodDeclaration) statement;
if (md.hasModifier(J.Modifier.Type.Native)) {
return cd;
}
}
}
if (checkFields.isEmpty()) {
return cd;
}
J.ClassDeclaration outer = cd;
for (Cursor parent = getCursor().getParent(); parent != null; parent = parent.getParent()) {
if (parent.getValue() instanceof J.ClassDeclaration) {
outer = parent.getValue();
}
}
for (CheckField checkField : checkFields) {
// Find variable uses.
Map> inUse =
VariableUses.find(checkField.declarations, outer);
for (Map.Entry> entry : inUse.entrySet()) {
if (entry.getValue().isEmpty()) {
AtomicBoolean declarationDeleted = new AtomicBoolean();
J.VariableDeclarations.NamedVariable fieldToRemove = entry.getKey();
cd = (J.ClassDeclaration) new RemoveUnusedField(fieldToRemove).visitNonNull(cd, declarationDeleted);
if (fieldToRemove.getType() != null) {
maybeRemoveImport(fieldToRemove.getType().toString());
}
// Maybe remove next statement comment if variable declarations is removed
if (declarationDeleted.get()) {
cd = (J.ClassDeclaration) new MaybeRemoveComment(checkField.nextStatement, cd).visitNonNull(cd, ctx);
}
}
}
}
return cd;
}
private boolean isSerialVersionUid(J.VariableDeclarations vd) {
return vd.hasModifier(J.Modifier.Type.Private) &&
vd.hasModifier(J.Modifier.Type.Static) &&
vd.hasModifier(J.Modifier.Type.Final) &&
TypeUtils.isOfClassType(vd.getType(), "long") &&
vd.getVariables().stream().anyMatch(it -> "serialVersionUID".equals(it.getSimpleName()));
}
};
}
private static class VariableUses {
public static Map> find(J.VariableDeclarations declarations, J.ClassDeclaration parent) {
Map> found = new IdentityHashMap<>(declarations.getVariables().size());
Map signatureMap = new HashMap<>();
for (J.VariableDeclarations.NamedVariable variable : declarations.getVariables()) {
if (variable.getVariableType() != null) {
found.computeIfAbsent(variable, k -> new ArrayList<>());
// Note: Using a variable type signature is only safe to find uses of class fields.
signatureMap.put(variable.getVariableType().toString(), variable);
}
}
JavaIsoVisitor