From 670b9ee637facd60af039b7b5df0c74c5e68e9af Mon Sep 17 00:00:00 2001 From: Liam Miller-Cushon Date: Thu, 20 Jun 2024 07:56:56 -0700 Subject: [PATCH] Convert some simple blocks to return switches using `yield` PiperOrigin-RevId: 645031086 --- .../StatementSwitchToExpressionSwitch.java | 48 ++++++++++++------- ...StatementSwitchToExpressionSwitchTest.java | 41 +++++++++++----- 2 files changed, 62 insertions(+), 27 deletions(-) diff --git a/core/src/main/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitch.java b/core/src/main/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitch.java index 824600c10159..ce273dc7587f 100644 --- a/core/src/main/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitch.java +++ b/core/src/main/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitch.java @@ -94,7 +94,7 @@ public final class StatementSwitchToExpressionSwitch extends BugChecker ImmutableSet.of(THROW, EXPRESSION_STATEMENT); private static final ImmutableSet KINDS_RETURN_OR_THROW = ImmutableSet.of(THROW, RETURN); private static final Pattern FALL_THROUGH_PATTERN = - Pattern.compile("\\bfalls?.?(through|out)\\b", Pattern.CASE_INSENSITIVE); + Pattern.compile("\\bfalls?.?through\\b", Pattern.CASE_INSENSITIVE); // Default (negative) result for assignment switch conversion analysis. Note that the value is // immutable. private static final AssignmentSwitchAnalysisResult DEFAULT_ASSIGNMENT_SWITCH_ANALYSIS_RESULT = @@ -325,14 +325,21 @@ private static CaseQualifications analyzeCaseForReturnSwitch( return previousCaseQualifications; } - // Statement blocks on the RHS are not currently supported - if (!(statements.size() == 1 && KINDS_RETURN_OR_THROW.contains(statements.get(0).getKind()))) { + // Statement blocks on the RHS are not currently supported, except for trivial blocks of + // statements expressions followed by a return or throw + // TODO: handle more complex statement blocks that can be converted using 'yield' + if (statements.isEmpty()) { + return CaseQualifications.SOME_OR_ALL_CASES_DONT_QUALIFY; + } + StatementTree lastStatement = getLast(statements); + if (!statements.subList(0, statements.size() - 1).stream() + .allMatch(statement -> statement.getKind().equals(EXPRESSION_STATEMENT)) + || !KINDS_RETURN_OR_THROW.contains(lastStatement.getKind())) { return CaseQualifications.SOME_OR_ALL_CASES_DONT_QUALIFY; } - StatementTree onlyStatement = statements.get(0); // For this analysis, cases that don't return something can be disregarded - if (!onlyStatement.getKind().equals(RETURN)) { + if (!lastStatement.getKind().equals(RETURN)) { return previousCaseQualifications; } @@ -343,7 +350,7 @@ private static CaseQualifications analyzeCaseForReturnSwitch( } // This is the first value-returning case that we are examining - Type returnType = ASTHelpers.getType(((ReturnTree) onlyStatement).getExpression()); + Type returnType = ASTHelpers.getType(((ReturnTree) lastStatement).getExpression()); return returnType == null // Return of void does not qualify ? CaseQualifications.SOME_OR_ALL_CASES_DONT_QUALIFY @@ -1117,21 +1124,30 @@ private static String transformReturnOrThrowBlock( CaseTree caseTree, VisitorState state, List statements) { StringBuilder transformedBlockBuilder = new StringBuilder(); - int codeBlockStart; - int codeBlockEnd = - statements.isEmpty() - ? getBlockEnd(state, caseTree) - : state.getEndPosition(Streams.findLast(statements.stream()).get()); - - if (statements.size() == 1 && statements.get(0).getKind().equals(RETURN)) { + int codeBlockEnd = state.getEndPosition(caseTree); + if (statements.size() > 1) { + transformedBlockBuilder.append("{\n"); + int codeBlockStart = extractLhsComments(caseTree, state, transformedBlockBuilder); + int offset = transformedBlockBuilder.length(); + transformedBlockBuilder.append(state.getSourceCode(), codeBlockStart, codeBlockEnd); + transformedBlockBuilder.append("\n}"); + ReturnTree returnTree = (ReturnTree) getLast(statements); + int start = getStartPosition(returnTree); + transformedBlockBuilder.replace( + offset + start - codeBlockStart, + offset + start - codeBlockStart + "return".length(), + "yield"); + } else if (statements.size() == 1 && statements.get(0).getKind().equals(RETURN)) { // For "return x;", we want to take source starting after the "return" int unused = extractLhsComments(caseTree, state, transformedBlockBuilder); ReturnTree returnTree = (ReturnTree) statements.get(0); - codeBlockStart = getStartPosition(returnTree.getExpression()); + int codeBlockStart = getStartPosition(returnTree.getExpression()); + codeBlockEnd = state.getEndPosition(Streams.findLast(statements.stream()).get()); + transformedBlockBuilder.append(state.getSourceCode(), codeBlockStart, codeBlockEnd); } else { - codeBlockStart = extractLhsComments(caseTree, state, transformedBlockBuilder); + int codeBlockStart = extractLhsComments(caseTree, state, transformedBlockBuilder); + transformedBlockBuilder.append(state.getSourceCode(), codeBlockStart, codeBlockEnd); } - transformedBlockBuilder.append(state.getSourceCode(), codeBlockStart, codeBlockEnd); return transformedBlockBuilder.toString(); } diff --git a/core/src/test/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitchTest.java b/core/src/test/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitchTest.java index 0ea13ecd6a18..461ae7928225 100644 --- a/core/src/test/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitchTest.java +++ b/core/src/test/java/com/google/errorprone/bugpatterns/StatementSwitchToExpressionSwitchTest.java @@ -3659,34 +3659,53 @@ public void unnecessaryBreaks() { } @Test - public void fallOutComment() { + public void mixedExpressionsAndYields() { assumeTrue(RuntimeVersion.isAtLeast14()); refactoringHelper .addInputLines( "Test.java", "public class Test {", - " void f(int x) {", + " String f(int x) {", " switch (x) {", " case 0:", - " System.err.println(\"ZERO\");", - " break;", + " return \"ZERO\";", + " case 1:", + " return \"ONE\";", + " case 2: // hello", + " // world", + " System.err.println();", + " System.err.println();", + " return \"TWO\";", + " // hello", + " // world", " default:", - " // fall out", + " return \"\";", " }", " }", "}") .addOutputLines( "Test.java", "public class Test {", - " void f(int x) {", - " switch (x) {", - " case 0 -> System.err.println(\"ZERO\");", - " default -> {}", - " }", + " String f(int x) {", + " return switch (x) {", + " case 0 -> \"ZERO\";", + " case 1 -> \"ONE\";", + " case 2 -> {", + " // hello", + " // world", + " System.err.println();", + " System.err.println();", + " yield \"TWO\";", + " }", + " // hello", + " // world", + " default -> \"\";", + " };", " }", "}") .setArgs( - ImmutableList.of("-XepOpt:StatementSwitchToExpressionSwitch:EnableDirectConversion")) + "-XepOpt:StatementSwitchToExpressionSwitch:EnableDirectConversion=true", + "-XepOpt:StatementSwitchToExpressionSwitch:EnableReturnSwitchConversion=true") .doTest(BugCheckerRefactoringTestHelper.TestMode.TEXT_MATCH); } }