-
Notifications
You must be signed in to change notification settings - Fork 39
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce EagerStringFormatting
check
#1139
Conversation
Quality Gate passedIssues Measures |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
1b69726
to
96c8208
Compare
0caa780
to
9675a51
Compare
Quality Gate passedIssues Measures |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
96c8208
to
8ee3ce8
Compare
9675a51
to
459da80
Compare
Quality Gate passedIssues Measures |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
8ee3ce8
to
4ce0955
Compare
3e30dc5
to
f200a92
Compare
459da80
to
49a376b
Compare
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
49a376b
to
edb6cc7
Compare
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
992e5fb
to
4bc7f8b
Compare
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
4bc7f8b
to
caff523
Compare
@rickie @mohamedsamehsalah this one is finally ready for review. :) |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧠 Saves a lot of review comments ❤️ ❤️
@Test | ||
void identification() { | ||
CompilationTestHelper.newInstance(EagerStringFormatting.class, getClass()) | ||
.expectErrorMessage("DEFER", m -> m.contains("String formatting can be deferred\n")) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is the \n
character necessary? 👀
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yep, cause otherwise this is a strictly weaker predicate than the DEFER_*
predicates below.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Makes sense 😄
import tech.picnic.errorprone.utils.SourceCode; | ||
|
||
/** | ||
* A {@link BugChecker} that flags {@link String#format(String, Object...)}, {@link |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe either just something along the lines of "String#format(String) and it's overloaded siblings", or format (heh) in a <ul>
tag.
(No strong opinion, this is also readable 👍 )
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I could do {@link String#format} and {@link String#formatted}
:)
import com.google.errorprone.CompilationTestHelper; | ||
import org.junit.jupiter.api.Test; | ||
|
||
final class EagerStringFormattingTest { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't we also check for flag the case when the potential String#format
invocations is stored and passed in a variable to the "parent" method?
Index: error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EagerStringFormattingTest.java
IDEA additional info:
Subsystem: com.intellij.openapi.diff.impl.patch.CharsetEP
<+>ISO-8859-1
===================================================================
diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EagerStringFormattingTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EagerStringFormattingTest.java
--- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EagerStringFormattingTest.java (revision caff523200614c14f23100fe4957dbf3d020bb4a)
+++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EagerStringFormattingTest.java (date 1735330311287)
@@ -65,6 +65,9 @@
"",
" requireNonNull(\"never-null\");",
" requireNonNull(\"never-null\", () -> String.format(\"Format string: %s\", nonFinalField));",
+ " // BUG: Diagnostic matches: DEFER_PARAM",
+ " String loggedMessage = String.format(\"Never-null format string: %s\", nonFinalField)",
+ " requireNonNull(loggedMessage);",
" // BUG: Diagnostic matches: VACUOUS",
" requireNonNull(String.format(\"Never-null format string: %s\", nonFinalField));",
" // BUG: Diagnostic matches: VACUOUS",
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is stored and passed in a variable to the "parent" method?
Or even go far and check if it's returned from another method invocation 👀
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indeed, none of these cases are currently covered. For now I'll call out these limitations in a comment; the check/PR is already complex enough as it is.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for taking such a quick look @mohamedsamehsalah! I added a commit.
import tech.picnic.errorprone.utils.SourceCode; | ||
|
||
/** | ||
* A {@link BugChecker} that flags {@link String#format(String, Object...)}, {@link |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I could do {@link String#format} and {@link String#formatted}
:)
@Test | ||
void identification() { | ||
CompilationTestHelper.newInstance(EagerStringFormatting.class, getClass()) | ||
.expectErrorMessage("DEFER", m -> m.contains("String formatting can be deferred\n")) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yep, cause otherwise this is a strictly weaker predicate than the DEFER_*
predicates below.
import com.google.errorprone.CompilationTestHelper; | ||
import org.junit.jupiter.api.Test; | ||
|
||
final class EagerStringFormattingTest { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indeed, none of these cases are currently covered. For now I'll call out these limitations in a comment; the check/PR is already complex enough as it is.
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧯
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Look at those IT 😍 , so nice! Flushing my two comments I have so far. It's a big one to get through with all the cases, but it will be such a nice one to release this.
linkType = CUSTOM, | ||
severity = WARNING, | ||
tags = {PERFORMANCE, SIMPLIFICATION}) | ||
public final class EagerStringFormatting extends BugChecker implements MethodInvocationTreeMatcher { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you elaborate on why we use "Eager" in the name of the check. In somewhat similar cases we use "Redundant", curious why we don't do that here :).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yep. Basically this check flags cases where string formatting happens unconditionally, rather than happening only if some other condition is met (precondition failure, log statement at enabled log level, ...). So the check flags "eager" formatting operations that could be "deferred" instead.
"", | ||
" checkArgument(true, String.format(\"Vacuous format string %%\"));", | ||
" checkNotNull(\"never-null\", \"Format string: %s %s%%\".formatted(1, 2));", | ||
" checkState(false, String.format(Locale.US, \"Format string with locale: %s\", 3));", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm curious, if we drop the Locale
, that could be non-behavior preserving, right?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIUC this should actually be fine, because we explicitly handle java.util.Formattable
arguments by not suggesting a rewrite when those are present.
This new check flags code that can be simplified and/or optimized by deferring certain string formatting operations.
3af3cf2
to
fa59f63
Compare
Quality Gate passedIssues Measures |
Mutation testing report by Pitest. Review any surviving mutants by inspecting the line comments under Files changed. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice work @Stephan202 , really curious to see this on our codebase!
❗ This PR is on top of #1138. ❗This is a draft PR; a bit more work (especially around tests) is required. I'm opening the PR already, as I'd like to reference it.
Suggested commit message: