From 1b8f663476bbad9de5226383245f52b6d9a96468 Mon Sep 17 00:00:00 2001 From: Stephan Schroevers Date: Fri, 8 Sep 2023 21:43:08 +0200 Subject: [PATCH] Support `@SuppressWarnings("all")` Checks that are suppressed using `@SuppressWarnings("CheckName")` are now also suppressed by `@SuppressWarnings("all")`, as emitted by libraries such as Lombok and Immutables. Checks that are unsuppressible, or can only be suppressed by a custom annotation are _not_ suppressed by `@SuppressWarnings("all")`. While there: improve the deprecated `BugChecker#isSuppressed(Tree)` and `BugChecker#isSuppressed(Symbol)` methods to respect `BugChecker#supportsSuppressWarnings()`. --- .../google/errorprone/SuppressionInfo.java | 3 +- .../errorprone/bugpatterns/BugChecker.java | 9 +- .../bugpatterns/BugCheckerTest.java | 182 ++++++++++++++++++ 3 files changed, 191 insertions(+), 3 deletions(-) create mode 100644 core/src/test/java/com/google/errorprone/bugpatterns/BugCheckerTest.java diff --git a/check_api/src/main/java/com/google/errorprone/SuppressionInfo.java b/check_api/src/main/java/com/google/errorprone/SuppressionInfo.java index cb23c08dea3..3b891b9cc5e 100644 --- a/check_api/src/main/java/com/google/errorprone/SuppressionInfo.java +++ b/check_api/src/main/java/com/google/errorprone/SuppressionInfo.java @@ -85,7 +85,8 @@ public SuppressedState suppressedState( return SuppressedState.SUPPRESSED; } if (suppressible.supportsSuppressWarnings() - && !Collections.disjoint(suppressible.allNames(), suppressWarningsStrings)) { + && (suppressWarningsStrings.contains("all") + || !Collections.disjoint(suppressible.allNames(), suppressWarningsStrings))) { return SuppressedState.SUPPRESSED; } if (suppressible.suppressedByAnyOf(customSuppressions, state)) { diff --git a/check_api/src/main/java/com/google/errorprone/bugpatterns/BugChecker.java b/check_api/src/main/java/com/google/errorprone/bugpatterns/BugChecker.java index 4693b70ef84..73cfa5b9584 100644 --- a/check_api/src/main/java/com/google/errorprone/bugpatterns/BugChecker.java +++ b/check_api/src/main/java/com/google/errorprone/bugpatterns/BugChecker.java @@ -99,6 +99,7 @@ import java.lang.annotation.Annotation; import java.util.Arrays; import java.util.Collections; +import java.util.List; import java.util.Objects; import java.util.Set; import java.util.function.BiPredicate; @@ -269,8 +270,12 @@ public boolean isSuppressed(Symbol symbol) { } private boolean isSuppressed(SuppressWarnings suppression) { - return suppression != null - && !Collections.disjoint(Arrays.asList(suppression.value()), allNames()); + if (suppression == null || !supportsSuppressWarnings()) { + return false; + } + + List suppressions = Arrays.asList(suppression.value()); + return suppressions.contains("all") || !Collections.disjoint(suppressions, allNames()); } /** diff --git a/core/src/test/java/com/google/errorprone/bugpatterns/BugCheckerTest.java b/core/src/test/java/com/google/errorprone/bugpatterns/BugCheckerTest.java new file mode 100644 index 00000000000..1213f9e3bb2 --- /dev/null +++ b/core/src/test/java/com/google/errorprone/bugpatterns/BugCheckerTest.java @@ -0,0 +1,182 @@ +/* + * Copyright 2023 The Error Prone 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 + * + * http://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 com.google.errorprone.bugpatterns; + +import static com.google.common.collect.ImmutableList.toImmutableList; +import static com.google.errorprone.BugPattern.SeverityLevel.ERROR; + +import com.google.common.collect.ImmutableList; +import com.google.errorprone.BugPattern; +import com.google.errorprone.CompilationTestHelper; +import com.google.errorprone.VisitorState; +import com.google.errorprone.bugpatterns.BugChecker.VariableTreeMatcher; +import com.google.errorprone.matchers.Description; +import com.sun.source.tree.VariableTree; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.JUnit4; + +@RunWith(JUnit4.class) +public class BugCheckerTest { + @Test + public void isSuppressed_withoutVisitorState() { + CompilationTestHelper.newInstance(LegacySuppressionCheck.class, getClass()) + .addSourceLines( + "A.java", + "class A {", + " void m() {", + " // BUG: Diagnostic contains: []", + " int unsuppressed;", + " // BUG: Diagnostic contains: []", + " @SuppressWarnings(\"foo\") int unrelatedSuppression;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings(\"Suppressible\") int suppressed;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings(\"Alternative\") int suppressedWithAlternativeName;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings(\"all\") int allSuppressed;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings({\"foo\", \"Suppressible\"}) int alsoSuppressed;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings({\"all\", \"foo\"}) int redundantlySuppressed;", + " // BUG: Diagnostic contains: [Suppressible]", + " @SuppressWarnings({\"all\", \"OnlySuppressedInsideDeprecatedCode\"}) int ineffectiveSuppression;", + " // BUG: Diagnostic contains: []", + " @Deprecated int unuspportedSuppression;", + " }", + "}") + .doTest(); + } + + @Test + public void isSuppressed() { + CompilationTestHelper.newInstance(SuppressibleCheck.class, getClass()) + .addSourceLines( + "A.java", + "class A {", + " void m() {", + " // BUG: Diagnostic contains:", + " int unsuppressed;", + " // BUG: Diagnostic contains:", + " @SuppressWarnings(\"foo\") int unrelatedSuppression;", + " @SuppressWarnings(\"Suppressible\") int suppressed;", + " @SuppressWarnings(\"Alternative\") int suppressedWithAlternativeName;", + " @SuppressWarnings(\"all\") int allSuppressed;", + " @SuppressWarnings({\"foo\", \"Suppressible\"}) int alsoSuppressed;", + " @SuppressWarnings({\"all\", \"foo\"}) int redundantlySuppressed;", + " }", + "}") + .doTest(); + } + + @Test + public void isSuppressed_customSuppressionAnnotation() { + CompilationTestHelper.newInstance(CustomSuppressibilityCheck.class, getClass()) + .addSourceLines( + "A.java", + "class A {", + " void m() {", + " // BUG: Diagnostic contains:", + " int unsuppressed;", + " // BUG: Diagnostic contains:", + " @SuppressWarnings({\"all\", \"OnlySuppressedInsideDeprecatedCode\"}) int ineffectiveSuppression;", + " @Deprecated int suppressed;", + " }", + "}") + .doTest(); + } + + @Test + public void isSuppressed_disableWarningsInGeneratedCode() { + CompilationTestHelper.newInstance(CustomSuppressibilityCheck.class, getClass()) + .setArgs("-XepDisableWarningsInGeneratedCode") + .addSourceLines( + "A.java", + "import javax.annotation.processing.Generated;", + "class A {", + " void m() {", + " // BUG: Diagnostic contains:", + " @Generated(\"some-tool\") int unsuppressed;", + " }", + "}") + .doTest(); + + // The check is suppressed if its severity is downgraded to `WARNING`. + CompilationTestHelper.newInstance(CustomSuppressibilityCheck.class, getClass()) + .setArgs( + "-XepDisableWarningsInGeneratedCode", "-Xep:OnlySuppressedInsideDeprecatedCode:WARN") + .addSourceLines( + "A.java", + "import javax.annotation.processing.Generated;", + "class A {", + " void m() {", + " @Generated(\"some-tool\") int unsuppressed;", + " }", + "}") + .doTest(); + } + + @BugPattern( + name = "SuppressionReporter", + summary = + "Tells whether some other checks are suppressed according to the deprecated method " + + "`BugChecker#isSuppressed(Tree)`", + severity = ERROR, + suppressionAnnotations = {}) + public static final class LegacySuppressionCheck extends BugChecker + implements VariableTreeMatcher { + private final ImmutableList checks = + ImmutableList.of(new SuppressibleCheck(), new CustomSuppressibilityCheck()); + + @Override + @SuppressWarnings("deprecation") // testing deprecated method + public Description matchVariable(VariableTree tree, VisitorState state) { + ImmutableList suppressions = + checks.stream() + .filter(check -> check.isSuppressed(tree)) + .map(BugChecker::canonicalName) + .collect(toImmutableList()); + + return buildDescription(tree).setMessage("Suppressions: " + suppressions).build(); + } + } + + @BugPattern( + name = "Suppressible", + altNames = "Alternative", + summary = "Can be suppressed", + severity = ERROR) + public static final class SuppressibleCheck extends BugChecker implements VariableTreeMatcher { + @Override + public Description matchVariable(VariableTree tree, VisitorState state) { + return describeMatch(tree); + } + } + + @BugPattern( + name = "OnlySuppressedInsideDeprecatedCode", + summary = "Can be suppressed using `@Deprecated`, but not `@SuppressWarnings`", + severity = ERROR, + suppressionAnnotations = Deprecated.class) + public static final class CustomSuppressibilityCheck extends BugChecker + implements VariableTreeMatcher { + @Override + public Description matchVariable(VariableTree tree, VisitorState state) { + return describeMatch(tree); + } + } +}