-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Introduce
OptionalOrElse
check (#1024)
While there, extend the `OptionalIdentity` Refaster rule to automatically resolve one class of `NestedOptionals` violations.
- Loading branch information
1 parent
e40df7e
commit 281a003
Showing
7 changed files
with
299 additions
and
16 deletions.
There are no files selected for viewing
135 changes: 135 additions & 0 deletions
135
error-prone-contrib/src/main/java/tech/picnic/errorprone/bugpatterns/OptionalOrElse.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,135 @@ | ||
package tech.picnic.errorprone.bugpatterns; | ||
|
||
import static com.google.errorprone.BugPattern.LinkType.NONE; | ||
import static com.google.errorprone.BugPattern.SeverityLevel.WARNING; | ||
import static com.google.errorprone.BugPattern.StandardTags.PERFORMANCE; | ||
import static com.google.errorprone.matchers.Matchers.instanceMethod; | ||
import static com.google.errorprone.matchers.Matchers.staticMethod; | ||
import static java.util.stream.Collectors.joining; | ||
|
||
import com.google.auto.service.AutoService; | ||
import com.google.common.collect.Iterables; | ||
import com.google.errorprone.BugPattern; | ||
import com.google.errorprone.VisitorState; | ||
import com.google.errorprone.bugpatterns.BugChecker; | ||
import com.google.errorprone.bugpatterns.BugChecker.MethodInvocationTreeMatcher; | ||
import com.google.errorprone.fixes.SuggestedFix; | ||
import com.google.errorprone.fixes.SuggestedFixes; | ||
import com.google.errorprone.matchers.Description; | ||
import com.google.errorprone.matchers.Matcher; | ||
import com.google.errorprone.refaster.Refaster; | ||
import com.google.errorprone.util.ASTHelpers; | ||
import com.sun.source.tree.ExpressionTree; | ||
import com.sun.source.tree.IdentifierTree; | ||
import com.sun.source.tree.LiteralTree; | ||
import com.sun.source.tree.MemberSelectTree; | ||
import com.sun.source.tree.MethodInvocationTree; | ||
import java.util.Optional; | ||
import java.util.function.Supplier; | ||
import tech.picnic.errorprone.utils.SourceCode; | ||
|
||
/** | ||
* A {@link BugChecker} that flags arguments to {@link Optional#orElse(Object)} that should be | ||
* deferred using {@link Optional#orElseGet(Supplier)}. | ||
* | ||
* <p>The suggested fix assumes that the argument to {@code orElse} does not have side effects. If | ||
* it does, the suggested fix changes the program's semantics. Such fragile code must instead be | ||
* refactored such that the side-effectful code does not appear accidental. | ||
*/ | ||
// XXX: Consider also implementing the inverse, in which `.orElseGet(() -> someConstant)` is | ||
// flagged. | ||
// XXX: Once the `MethodReferenceUsageCheck` becomes generally usable, consider leaving the method | ||
// reference cleanup to that check, and express the remainder of the logic in this class using a | ||
// Refaster template, i.c.w. a `@Matches` constraint that implements the `requiresComputation` | ||
// logic. | ||
@AutoService(BugChecker.class) | ||
@BugPattern( | ||
summary = | ||
""" | ||
Prefer `Optional#orElseGet` over `Optional#orElse` if the fallback requires additional \ | ||
computation""", | ||
linkType = NONE, | ||
severity = WARNING, | ||
tags = PERFORMANCE) | ||
public final class OptionalOrElse extends BugChecker implements MethodInvocationTreeMatcher { | ||
private static final long serialVersionUID = 1L; | ||
private static final Matcher<ExpressionTree> OPTIONAL_OR_ELSE_METHOD = | ||
instanceMethod().onExactClass(Optional.class.getCanonicalName()).namedAnyOf("orElse"); | ||
// XXX: Also exclude invocations of `@Placeholder`-annotated methods. | ||
private static final Matcher<ExpressionTree> REFASTER_METHOD = | ||
staticMethod().onClass(Refaster.class.getCanonicalName()); | ||
|
||
/** Instantiates a new {@link OptionalOrElse} instance. */ | ||
public OptionalOrElse() {} | ||
|
||
@Override | ||
public Description matchMethodInvocation(MethodInvocationTree tree, VisitorState state) { | ||
if (!OPTIONAL_OR_ELSE_METHOD.matches(tree, state)) { | ||
return Description.NO_MATCH; | ||
} | ||
|
||
ExpressionTree argument = Iterables.getOnlyElement(tree.getArguments()); | ||
if (!requiresComputation(argument) || REFASTER_METHOD.matches(argument, state)) { | ||
return Description.NO_MATCH; | ||
} | ||
|
||
/* | ||
* We have a match. Construct the method reference or lambda expression to be passed to the | ||
* replacement `#orElseGet` invocation. | ||
*/ | ||
String newArgument = | ||
tryMethodReferenceConversion(argument, state) | ||
.orElseGet(() -> "() -> " + SourceCode.treeToString(argument, state)); | ||
|
||
/* Construct the suggested fix, replacing the method invocation and its argument. */ | ||
SuggestedFix fix = | ||
SuggestedFix.builder() | ||
.merge(SuggestedFixes.renameMethodInvocation(tree, "orElseGet", state)) | ||
.replace(argument, newArgument) | ||
.build(); | ||
|
||
return describeMatch(tree, fix); | ||
} | ||
|
||
/** | ||
* Tells whether the given expression contains anything other than a literal or a (possibly | ||
* dereferenced) variable or constant. | ||
*/ | ||
private static boolean requiresComputation(ExpressionTree tree) { | ||
return !(tree instanceof IdentifierTree | ||
|| tree instanceof LiteralTree | ||
|| (tree instanceof MemberSelectTree memberSelect | ||
&& !requiresComputation(memberSelect.getExpression())) | ||
|| ASTHelpers.constValue(tree) != null); | ||
} | ||
|
||
/** Returns the nullary method reference matching the given expression, if any. */ | ||
private static Optional<String> tryMethodReferenceConversion( | ||
ExpressionTree tree, VisitorState state) { | ||
if (!(tree instanceof MethodInvocationTree methodInvocation)) { | ||
return Optional.empty(); | ||
} | ||
|
||
if (!methodInvocation.getArguments().isEmpty()) { | ||
return Optional.empty(); | ||
} | ||
|
||
if (!(methodInvocation.getMethodSelect() instanceof MemberSelectTree memberSelect)) { | ||
return Optional.empty(); | ||
} | ||
|
||
if (requiresComputation(memberSelect.getExpression())) { | ||
return Optional.empty(); | ||
} | ||
|
||
return Optional.of( | ||
SourceCode.treeToString(memberSelect.getExpression(), state) | ||
+ "::" | ||
+ (methodInvocation.getTypeArguments().isEmpty() | ||
? "" | ||
: methodInvocation.getTypeArguments().stream() | ||
.map(arg -> SourceCode.treeToString(arg, state)) | ||
.collect(joining(",", "<", ">"))) | ||
+ memberSelect.getIdentifier()); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
135 changes: 135 additions & 0 deletions
135
error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/OptionalOrElseTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,135 @@ | ||
package tech.picnic.errorprone.bugpatterns; | ||
|
||
import com.google.errorprone.BugCheckerRefactoringTestHelper; | ||
import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode; | ||
import com.google.errorprone.CompilationTestHelper; | ||
import org.junit.jupiter.api.Test; | ||
|
||
final class OptionalOrElseTest { | ||
@Test | ||
void identification() { | ||
CompilationTestHelper.newInstance(OptionalOrElse.class, getClass()) | ||
.addSourceLines( | ||
"A.java", | ||
"import com.google.errorprone.refaster.Refaster;", | ||
"import java.util.Optional;", | ||
"", | ||
"class A {", | ||
" private final Optional<Object> optional = Optional.empty();", | ||
" private final String string = optional.toString();", | ||
"", | ||
" void m() {", | ||
" Optional.empty().orElse(null);", | ||
" optional.orElse(null);", | ||
" optional.orElse(\"constant\");", | ||
" optional.orElse(\"constant\" + 0);", | ||
" optional.orElse(Boolean.TRUE);", | ||
" optional.orElse(string);", | ||
" optional.orElse(this.string);", | ||
" optional.orElse(Refaster.anyOf(\"constant\", \"another\"));", | ||
"", | ||
" // BUG: Diagnostic contains:", | ||
" Optional.empty().orElse(string + \"constant\");", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(string + \"constant\");", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(\"constant\".toString());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(string.toString());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(this.string.toString());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(String.valueOf(42));", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(string.toString().length());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(\"constant\".equals(string));", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(string.equals(string));", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(this.string.equals(string));", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(foo());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(this.foo());", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(new Object() {});", | ||
" // BUG: Diagnostic contains:", | ||
" optional.orElse(new int[0].length);", | ||
" }", | ||
"", | ||
" private <T> T foo() {", | ||
" return null;", | ||
" }", | ||
"}") | ||
.doTest(); | ||
} | ||
|
||
@Test | ||
void replacement() { | ||
BugCheckerRefactoringTestHelper.newInstance(OptionalOrElse.class, getClass()) | ||
.addInputLines( | ||
"A.java", | ||
"import java.util.Optional;", | ||
"", | ||
"class A {", | ||
" private final Optional<Object> optional = Optional.empty();", | ||
" private final String string = optional.toString();", | ||
"", | ||
" void m() {", | ||
" optional.orElse(string + \"constant\");", | ||
" optional.orElse(\"constant\".toString());", | ||
" optional.orElse(string.toString());", | ||
" optional.orElse(this.string.toString());", | ||
" optional.orElse(String.valueOf(42));", | ||
" optional.orElse(string.toString().length());", | ||
" optional.orElse(string.equals(string));", | ||
" optional.orElse(foo());", | ||
" optional.orElse(this.<Number>foo());", | ||
" optional.orElse(this.<String, Integer>bar());", | ||
" optional.orElse(new Object() {});", | ||
" optional.orElse(new int[0].length);", | ||
" }", | ||
"", | ||
" private <T> T foo() {", | ||
" return null;", | ||
" }", | ||
"", | ||
" private <S, T> T bar() {", | ||
" return null;", | ||
" }", | ||
"}") | ||
.addOutputLines( | ||
"A.java", | ||
"import java.util.Optional;", | ||
"", | ||
"class A {", | ||
" private final Optional<Object> optional = Optional.empty();", | ||
" private final String string = optional.toString();", | ||
"", | ||
" void m() {", | ||
" optional.orElseGet(() -> string + \"constant\");", | ||
" optional.orElseGet(\"constant\"::toString);", | ||
" optional.orElseGet(string::toString);", | ||
" optional.orElseGet(this.string::toString);", | ||
" optional.orElseGet(() -> String.valueOf(42));", | ||
" optional.orElseGet(() -> string.toString().length());", | ||
" optional.orElseGet(() -> string.equals(string));", | ||
" optional.orElseGet(() -> foo());", | ||
" optional.orElseGet(this::<Number>foo);", | ||
" optional.orElseGet(this::<String, Integer>bar);", | ||
" optional.orElseGet(() -> new Object() {});", | ||
" optional.orElseGet(() -> new int[0].length);", | ||
" }", | ||
"", | ||
" private <T> T foo() {", | ||
" return null;", | ||
" }", | ||
"", | ||
" private <S, T> T bar() {", | ||
" return null;", | ||
" }", | ||
"}") | ||
.doTest(TestMode.TEXT_MATCH); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters