From 81e3cfa0113f2c9d19ae8cfd4f2d33b52cffa7bd Mon Sep 17 00:00:00 2001 From: Stephan Schroevers Date: Wed, 30 Oct 2024 08:18:02 -0700 Subject: [PATCH] Introduce `IdentifierName:AllowInitialismsInTypeName` flag While default `IdentifierName` behavior is unchanged, this flag allows users to slightly relax the type name validation performed by this check. Fixes #4646 COPYBARA_INTEGRATE_REVIEW=https://github.com/google/error-prone/pull/4646 from PicnicSupermarket:sschroevers/allow-initialisms-in-type-names 82abeaf229a11e1f6211c3fac170c811359e8210 PiperOrigin-RevId: 691425146 --- .../errorprone/bugpatterns/IdentifierName.java | 18 ++++++++++++++---- .../bugpatterns/IdentifierNameTest.java | 11 +++++++++++ 2 files changed, 25 insertions(+), 4 deletions(-) diff --git a/core/src/main/java/com/google/errorprone/bugpatterns/IdentifierName.java b/core/src/main/java/com/google/errorprone/bugpatterns/IdentifierName.java index a34c5f1ef3b..9cd1344d113 100644 --- a/core/src/main/java/com/google/errorprone/bugpatterns/IdentifierName.java +++ b/core/src/main/java/com/google/errorprone/bugpatterns/IdentifierName.java @@ -41,6 +41,7 @@ import com.google.common.base.Splitter; import com.google.common.collect.ImmutableSet; import com.google.errorprone.BugPattern; +import com.google.errorprone.ErrorProneFlags; import com.google.errorprone.VisitorState; import com.google.errorprone.bugpatterns.BugChecker.ClassTreeMatcher; import com.google.errorprone.bugpatterns.BugChecker.MethodTreeMatcher; @@ -60,6 +61,7 @@ import com.sun.tools.javac.util.Name; import java.util.regex.Pattern; import java.util.stream.Stream; +import javax.inject.Inject; import javax.lang.model.element.ElementKind; import javax.lang.model.element.Modifier; @@ -98,17 +100,25 @@ public final class IdentifierName extends BugChecker ", with acronyms treated as words" + " (https://google.github.io/styleguide/javaguide.html#s5.3-camel-case)"; + private final boolean allowInitialismsInTypeName; + + @Inject + IdentifierName(ErrorProneFlags flags) { + this.allowInitialismsInTypeName = + flags.getBoolean("IdentifierName:AllowInitialismsInTypeName").orElse(false); + } + @Override public Description matchClass(ClassTree tree, VisitorState state) { ClassSymbol symbol = getSymbol(tree); String name = tree.getSimpleName().toString(); - if (name.isEmpty() || isConformantUpperCamelName(name)) { + if (name.isEmpty() || isConformantTypeName(name)) { // The name can be empty for enum member declarations, which are desugared early to class // declarations. return NO_MATCH; } String renamed = suggestedClassRename(name); - String suggested = fixInitialisms(renamed); + String suggested = allowInitialismsInTypeName ? renamed : fixInitialisms(renamed); boolean fixable = !suggested.equals(name) && canBeRemoved(symbol); String diagnostic = "Classes should be named in UpperCamelCase" @@ -281,10 +291,10 @@ private static boolean isConformantLowerCamelName(String name) { && !PROBABLE_INITIALISM.matcher(name).find(); } - private static boolean isConformantUpperCamelName(String name) { + private boolean isConformantTypeName(String name) { return !name.contains("_") && isUpperCase(name.charAt(0)) - && !PROBABLE_INITIALISM.matcher(name).find(); + && (allowInitialismsInTypeName || !PROBABLE_INITIALISM.matcher(name).find()); } private static boolean isStaticVariable(Symbol symbol) { diff --git a/core/src/test/java/com/google/errorprone/bugpatterns/IdentifierNameTest.java b/core/src/test/java/com/google/errorprone/bugpatterns/IdentifierNameTest.java index 31d3f0ec22a..a748abf723b 100644 --- a/core/src/test/java/com/google/errorprone/bugpatterns/IdentifierNameTest.java +++ b/core/src/test/java/com/google/errorprone/bugpatterns/IdentifierNameTest.java @@ -601,6 +601,17 @@ public void className_badInitialism() { .doTest(); } + @Test + public void className_badInitialism_allowed() { + helper + .setArgs("-XepOpt:IdentifierName:AllowInitialismsInTypeName=true") + .addSourceLines( + "Test.java", // + "class RPCServiceTester {", + "}") + .doTest(); + } + @Test public void className_lowerCamelCase() { helper