From 201ea7c3ede010d6a5ed9d57b26fd686b46bde56 Mon Sep 17 00:00:00 2001 From: Stephan Schroevers Date: Sun, 14 Aug 2022 21:38:46 +0200 Subject: [PATCH] Use text blocks (but with excess newlines; fix) --- .../bugpatterns/AmbiguousJsonCreatorTest.java | 235 +-- .../bugpatterns/AssertJIsNullTest.java | 73 +- .../bugpatterns/AutowiredConstructorTest.java | 165 +- .../CanonicalAnnotationSyntaxTest.java | 495 +++--- .../bugpatterns/CollectorMutabilityTest.java | 327 ++-- .../bugpatterns/EmptyMethodTest.java | 116 +- .../ErrorProneTestHelperSourceFormatTest.java | 370 ++--- .../bugpatterns/ExplicitEnumOrderingTest.java | 139 +- .../bugpatterns/FluxFlatMapUsageTest.java | 169 +- .../FormatStringConcatenationTest.java | 773 ++++----- .../bugpatterns/IdentityConversionTest.java | 531 ++++--- .../JUnitMethodDeclarationTest.java | 789 +++++----- ...aphicalAnnotationAttributeListingTest.java | 474 +++--- .../LexicographicalAnnotationListingTest.java | 421 ++--- .../bugpatterns/MethodReferenceUsageTest.java | 819 +++++----- .../MissingRefasterAnnotationTest.java | 143 +- .../bugpatterns/MockitoStubbingTest.java | 157 +- .../bugpatterns/PrimitiveComparisonTest.java | 1388 +++++++++-------- .../RedundantStringConversionTest.java | 872 ++++++----- .../bugpatterns/RefasterAnyOfUsageTest.java | 83 +- .../RequestMappingAnnotationTest.java | 233 +-- .../bugpatterns/RequestParamTypeTest.java | 101 +- .../ScheduledTransactionTraceTest.java | 135 +- .../bugpatterns/Slf4JLogStatementTest.java | 241 +-- .../bugpatterns/SpringMvcAnnotationTest.java | 253 +-- .../bugpatterns/StaticImportTest.java | 455 +++--- .../bugpatterns/TimeZoneUsageTest.java | 163 +- .../util/MethodMatcherFactoryTest.java | 292 ++-- .../errorprone/refaster/util/IsArrayTest.java | 67 +- 29 files changed, 5406 insertions(+), 5073 deletions(-) diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AmbiguousJsonCreatorTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AmbiguousJsonCreatorTest.java index a59bb2e7f02..58017cc3214 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AmbiguousJsonCreatorTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AmbiguousJsonCreatorTest.java @@ -20,99 +20,102 @@ void identification() { compilationTestHelper .addSourceLines( "Container.java", - "import com.fasterxml.jackson.annotation.JsonCreator;", - "import com.fasterxml.jackson.annotation.JsonValue;", - "", - "interface Container {", - " enum A {", - " FOO(1);", - "", - " private final int i;", - "", - " A(int i) {", - " this.i = i;", - " }", - "", - " // BUG: Diagnostic matches: X", - " @JsonCreator", - " public static A of(int i) {", - " return FOO;", - " }", - " }", - "", - " enum B {", - " FOO(1);", - "", - " private final int i;", - "", - " B(int i) {", - " this.i = i;", - " }", - "", - " @JsonCreator(mode = JsonCreator.Mode.DELEGATING)", - " public static B of(int i) {", - " return FOO;", - " }", - " }", - "", - " enum C {", - " FOO(1, \"s\");", - "", - " @JsonValue private final int i;", - " private final String s;", - "", - " C(int i, String s) {", - " this.i = i;", - " this.s = s;", - " }", - "", - " // BUG: Diagnostic matches: X", - " @JsonCreator", - " public static C of(int i) {", - " return FOO;", - " }", - " }", - "", - " enum D {", - " FOO(1, \"s\");", - "", - " private final int i;", - " private final String s;", - "", - " D(int i, String s) {", - " this.i = i;", - " this.s = s;", - " }", - "", - " @JsonCreator", - " public static D of(int i, String s) {", - " return FOO;", - " }", - " }", - "", - " enum E {", - " FOO;", - "", - " // BUG: Diagnostic matches: X", - " @JsonCreator", - " public static E of(String s) {", - " return FOO;", - " }", - " }", - "", - " class F {", - " private final String s;", - "", - " F(String s) {", - " this.s = s;", - " }", - "", - " @JsonCreator", - " public static F of(String s) {", - " return new F(s);", - " }", - " }", - "}") + """ + import com.fasterxml.jackson.annotation.JsonCreator; + import com.fasterxml.jackson.annotation.JsonValue; + + interface Container { + enum A { + FOO(1); + + private final int i; + + A(int i) { + this.i = i; + } + + // BUG: Diagnostic matches: X + @JsonCreator + public static A of(int i) { + return FOO; + } + } + + enum B { + FOO(1); + + private final int i; + + B(int i) { + this.i = i; + } + + @JsonCreator(mode = JsonCreator.Mode.DELEGATING) + public static B of(int i) { + return FOO; + } + } + + enum C { + FOO(1, "s"); + + @JsonValue private final int i; + private final String s; + + C(int i, String s) { + this.i = i; + this.s = s; + } + + // BUG: Diagnostic matches: X + @JsonCreator + public static C of(int i) { + return FOO; + } + } + + enum D { + FOO(1, "s"); + + private final int i; + private final String s; + + D(int i, String s) { + this.i = i; + this.s = s; + } + + @JsonCreator + public static D of(int i, String s) { + return FOO; + } + } + + enum E { + FOO; + + // BUG: Diagnostic matches: X + @JsonCreator + public static E of(String s) { + return FOO; + } + } + + class F { + private final String s; + + F(String s) { + this.s = s; + } + + @JsonCreator + public static F of(String s) { + return new F(s); + } + } + } + + """) .doTest(); } @@ -121,28 +124,34 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import com.fasterxml.jackson.annotation.JsonCreator;", - "", - "enum A {", - " FOO;", - "", - " @JsonCreator", - " public static A of(String s) {", - " return FOO;", - " }", - "}") + """ + import com.fasterxml.jackson.annotation.JsonCreator; + + enum A { + FOO; + + @JsonCreator + public static A of(String s) { + return FOO; + } + } + + """) .addOutputLines( "out/A.java", - "import com.fasterxml.jackson.annotation.JsonCreator;", - "", - "enum A {", - " FOO;", - "", - " @JsonCreator(mode = JsonCreator.Mode.DELEGATING)", - " public static A of(String s) {", - " return FOO;", - " }", - "}") + """ + import com.fasterxml.jackson.annotation.JsonCreator; + + enum A { + FOO; + + @JsonCreator(mode = JsonCreator.Mode.DELEGATING) + public static A of(String s) { + return FOO; + } + } + + """) .doTest(BugCheckerRefactoringTestHelper.TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AssertJIsNullTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AssertJIsNullTest.java index a0f8242dfb4..75ceb82b225 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AssertJIsNullTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AssertJIsNullTest.java @@ -17,22 +17,25 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static org.assertj.core.api.Assertions.assertThat;", - "", - "class A {", - " void m() {", - " assertThat(1).isEqualTo(1);", - " // BUG: Diagnostic contains:", - " assertThat(1).isEqualTo(null);", - " // BUG: Diagnostic contains:", - " assertThat(\"foo\").isEqualTo(null);", - " isEqualTo(null);", - " }", - "", - " private boolean isEqualTo(Object value) {", - " return value.equals(\"bar\");", - " }", - "}") + """ + import static org.assertj.core.api.Assertions.assertThat; + + class A { + void m() { + assertThat(1).isEqualTo(1); + // BUG: Diagnostic contains: + assertThat(1).isEqualTo(null); + // BUG: Diagnostic contains: + assertThat("foo").isEqualTo(null); + isEqualTo(null); + } + + private boolean isEqualTo(Object value) { + return value.equals("bar"); + } + } + + """) .doTest(); } @@ -41,24 +44,30 @@ void replacement() { refactoringTestHelper .addInputLines( "A.java", - "import static org.assertj.core.api.Assertions.assertThat;", - "", - "class A {", - " void m() {", - " assertThat(1).isEqualTo(null);", - " assertThat(\"foo\").isEqualTo(null);", - " }", - "}") + """ + import static org.assertj.core.api.Assertions.assertThat; + + class A { + void m() { + assertThat(1).isEqualTo(null); + assertThat("foo").isEqualTo(null); + } + } + + """) .addOutputLines( "A.java", - "import static org.assertj.core.api.Assertions.assertThat;", - "", - "class A {", - " void m() {", - " assertThat(1).isNull();", - " assertThat(\"foo\").isNull();", - " }", - "}") + """ + import static org.assertj.core.api.Assertions.assertThat; + + class A { + void m() { + assertThat(1).isNull(); + assertThat("foo").isNull(); + } + } + + """) .doTest(TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AutowiredConstructorTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AutowiredConstructorTest.java index 0e5916892bc..40e67327415 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AutowiredConstructorTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/AutowiredConstructorTest.java @@ -16,56 +16,59 @@ void identification() { compilationTestHelper .addSourceLines( "Container.java", - "import com.google.errorprone.annotations.Immutable;", - "import java.util.List;", - "import org.springframework.beans.factory.annotation.Autowired;", - "", - "interface Container {", - " @Immutable", - " class A {", - " A() {}", - " }", - "", - " class B {", - " @Autowired", - " void setProperty(Object o) {}", - " }", - "", - " class C {", - " // BUG: Diagnostic contains:", - " @Autowired", - " C() {}", - " }", - "", - " class D {", - " // BUG: Diagnostic contains:", - " @Autowired", - " D(String x) {}", - " }", - "", - " class E {", - " @Autowired", - " E() {}", - "", - " E(String x) {}", - " }", - "", - " class F {", - " F() {}", - "", - " @Autowired", - " F(String x) {}", - " }", - "", - " class G {", - " @Autowired private Object o;", - " }", - "", - " class H {", - " @SafeVarargs", - " H(List... lists) {}", - " }", - "}") + """ + import com.google.errorprone.annotations.Immutable; + import java.util.List; + import org.springframework.beans.factory.annotation.Autowired; + + interface Container { + @Immutable + class A { + A() {} + } + + class B { + @Autowired + void setProperty(Object o) {} + } + + class C { + // BUG: Diagnostic contains: + @Autowired + C() {} + } + + class D { + // BUG: Diagnostic contains: + @Autowired + D(String x) {} + } + + class E { + @Autowired + E() {} + + E(String x) {} + } + + class F { + F() {} + + @Autowired + F(String x) {} + } + + class G { + @Autowired private Object o; + } + + class H { + @SafeVarargs + H(List... lists) {} + } + } + + """) .doTest(); } @@ -74,36 +77,42 @@ void replacement() { refactoringTestHelper .addInputLines( "in/Container.java", - "import org.springframework.beans.factory.annotation.Autowired;", - "", - "interface Container {", - " class A {", - " @Autowired", - " @Deprecated", - " A() {}", - " }", - "", - " class B {", - " @Autowired", - " B(String x) {}", - " }", - "}") + """ + import org.springframework.beans.factory.annotation.Autowired; + + interface Container { + class A { + @Autowired + @Deprecated + A() {} + } + + class B { + @Autowired + B(String x) {} + } + } + + """) .addOutputLines( "out/Container.java", - "import org.springframework.beans.factory.annotation.Autowired;", - "", - "interface Container {", - " class A {", - "", - " @Deprecated", - " A() {}", - " }", - "", - " class B {", - "", - " B(String x) {}", - " }", - "}") + """ + import org.springframework.beans.factory.annotation.Autowired; + + interface Container { + class A { + + @Deprecated + A() {} + } + + class B { + + B(String x) {} + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CanonicalAnnotationSyntaxTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CanonicalAnnotationSyntaxTest.java index a6ae6c673af..f590086fea0 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CanonicalAnnotationSyntaxTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CanonicalAnnotationSyntaxTest.java @@ -16,118 +16,121 @@ void identification() { compilationTestHelper .addSourceLines( "pkg/A.java", - "package pkg;", - "", - "import pkg.A.Foo;", - "", - "interface A {", - " @interface Foo {", - " int[] value() default {};", - "", - " int[] value2() default {};", - " }", - "", - " @pkg.A.Foo", - " A minimal1();", - "", - " @A.Foo", - " A minimal2();", - "", - " @Foo", - " A minimal3();", - "", - " // BUG: Diagnostic contains:", - " @pkg.A.Foo()", - " A functional1();", - " // BUG: Diagnostic contains:", - " @A.Foo()", - " A functional2();", - " // BUG: Diagnostic contains:", - " @Foo()", - " A functional3();", - "", - " @pkg.A.Foo(1)", - " A simple1();", - "", - " @A.Foo(1)", - " A simple2();", - "", - " @Foo(1)", - " A simple3();", - "", - " // BUG: Diagnostic contains:", - " @pkg.A.Foo({1})", - " A singleton1();", - " // BUG: Diagnostic contains:", - " @A.Foo({1})", - " A singleton2();", - " // BUG: Diagnostic contains:", - " @Foo({1})", - " A singleton3();", - "", - " // BUG: Diagnostic contains:", - " @pkg.A.Foo(value = 1)", - " A verbose1();", - " // BUG: Diagnostic contains:", - " @A.Foo(value = 1)", - " A verbose2();", - " // BUG: Diagnostic contains:", - " @Foo(value = 1)", - " A verbose3();", - "", - " @pkg.A.Foo(value2 = 2)", - " A custom1();", - "", - " @A.Foo(value2 = 2)", - " A custom2();", - "", - " @Foo(value2 = 2)", - " A custom3();", - "", - " // BUG: Diagnostic contains:", - " @pkg.A.Foo(value2 = {2})", - " A customSingleton1();", - " // BUG: Diagnostic contains:", - " @A.Foo(value2 = {2})", - " A customSingleton2();", - " // BUG: Diagnostic contains:", - " @Foo(value2 = {2})", - " A customSingleton3();", - "", - " @pkg.A.Foo(value2 = {2, 2})", - " A customPair1();", - "", - " @A.Foo(value2 = {2, 2})", - " A customPair2();", - "", - " @Foo(value2 = {2, 2})", - " A customPair3();", - "", - " @pkg.A.Foo(value = 1, value2 = 2)", - " A extended1();", - "", - " @A.Foo(value = 1, value2 = 2)", - " A extended2();", - "", - " @Foo(value = 1, value2 = 2)", - " A extended3();", - "", - " // BUG: Diagnostic contains:", - " @pkg.A.Foo({", - " 1, 1,", - " })", - " A trailingComma1();", - " // BUG: Diagnostic contains:", - " @A.Foo({", - " 1, 1,", - " })", - " A trailingComma2();", - " // BUG: Diagnostic contains:", - " @Foo({", - " 1, 1,", - " })", - " A trailingComma3();", - "}") + """ + package pkg; + + import pkg.A.Foo; + + interface A { + @interface Foo { + int[] value() default {}; + + int[] value2() default {}; + } + + @pkg.A.Foo + A minimal1(); + + @A.Foo + A minimal2(); + + @Foo + A minimal3(); + + // BUG: Diagnostic contains: + @pkg.A.Foo() + A functional1(); + // BUG: Diagnostic contains: + @A.Foo() + A functional2(); + // BUG: Diagnostic contains: + @Foo() + A functional3(); + + @pkg.A.Foo(1) + A simple1(); + + @A.Foo(1) + A simple2(); + + @Foo(1) + A simple3(); + + // BUG: Diagnostic contains: + @pkg.A.Foo({1}) + A singleton1(); + // BUG: Diagnostic contains: + @A.Foo({1}) + A singleton2(); + // BUG: Diagnostic contains: + @Foo({1}) + A singleton3(); + + // BUG: Diagnostic contains: + @pkg.A.Foo(value = 1) + A verbose1(); + // BUG: Diagnostic contains: + @A.Foo(value = 1) + A verbose2(); + // BUG: Diagnostic contains: + @Foo(value = 1) + A verbose3(); + + @pkg.A.Foo(value2 = 2) + A custom1(); + + @A.Foo(value2 = 2) + A custom2(); + + @Foo(value2 = 2) + A custom3(); + + // BUG: Diagnostic contains: + @pkg.A.Foo(value2 = {2}) + A customSingleton1(); + // BUG: Diagnostic contains: + @A.Foo(value2 = {2}) + A customSingleton2(); + // BUG: Diagnostic contains: + @Foo(value2 = {2}) + A customSingleton3(); + + @pkg.A.Foo(value2 = {2, 2}) + A customPair1(); + + @A.Foo(value2 = {2, 2}) + A customPair2(); + + @Foo(value2 = {2, 2}) + A customPair3(); + + @pkg.A.Foo(value = 1, value2 = 2) + A extended1(); + + @A.Foo(value = 1, value2 = 2) + A extended2(); + + @Foo(value = 1, value2 = 2) + A extended3(); + + // BUG: Diagnostic contains: + @pkg.A.Foo({ + 1, 1, + }) + A trailingComma1(); + // BUG: Diagnostic contains: + @A.Foo({ + 1, 1, + }) + A trailingComma2(); + // BUG: Diagnostic contains: + @Foo({ + 1, 1, + }) + A trailingComma3(); + } + + """) .doTest(); } @@ -136,139 +139,145 @@ void replacement() { refactoringTestHelper .addInputLines( "in/pkg/A.java", - "package pkg;", - "", - "import pkg.A.Foo;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] value2() default {};", - " }", - "", - " @pkg.A.Foo()", - " A functional1();", - "", - " @A.Foo()", - " A functional2();", - "", - " @Foo()", - " A functional3();", - "", - " @pkg.A.Foo(value = \"foo\")", - " A verbose1();", - "", - " @A.Foo(value = \"a'b\")", - " A verbose2();", - "", - " @Foo(value = \"a\" + \"\\nb\")", - " A verbose3();", - "", - " @pkg.A.Foo(value = {\"foo\"})", - " A moreVerbose1();", - "", - " @A.Foo(value = {\"a'b\"})", - " A moreVerbose2();", - "", - " @Foo(value = {\"a\" + \"\\nb\"})", - " A moreVerbose3();", - "", - " @pkg.A.Foo(", - " value = {\"foo\", \"bar\"},", - " value2 = {2})", - " A extended1();", - "", - " @A.Foo(", - " value = {\"a'b\", \"c'd\"},", - " value2 = {2})", - " A extended2();", - "", - " @Foo(", - " value = {\"a\" + \"\\nb\", \"c\" + \"\\nd\"},", - " value2 = {2})", - " A extended3();", - "", - " @pkg.A.Foo({", - " \"foo\", \"bar\",", - " })", - " A trailingComma1();", - "", - " @A.Foo({", - " \"a'b\", \"c'd\",", - " })", - " A trailingComma2();", - "", - " @Foo({", - " \"a\" + \"\\nb\",", - " \"c\" + \"\\nd\",", - " })", - " A trailingComma3();", - "}") + """ + package pkg; + + import pkg.A.Foo; + + interface A { + @interface Foo { + String[] value() default {}; + + int[] value2() default {}; + } + + @pkg.A.Foo() + A functional1(); + + @A.Foo() + A functional2(); + + @Foo() + A functional3(); + + @pkg.A.Foo(value = "foo") + A verbose1(); + + @A.Foo(value = "a'b") + A verbose2(); + + @Foo(value = "a" + "\\nb") + A verbose3(); + + @pkg.A.Foo(value = {"foo"}) + A moreVerbose1(); + + @A.Foo(value = {"a'b"}) + A moreVerbose2(); + + @Foo(value = {"a" + "\\nb"}) + A moreVerbose3(); + + @pkg.A.Foo( + value = {"foo", "bar"}, + value2 = {2}) + A extended1(); + + @A.Foo( + value = {"a'b", "c'd"}, + value2 = {2}) + A extended2(); + + @Foo( + value = {"a" + "\\nb", "c" + "\\nd"}, + value2 = {2}) + A extended3(); + + @pkg.A.Foo({ + "foo", "bar", + }) + A trailingComma1(); + + @A.Foo({ + "a'b", "c'd", + }) + A trailingComma2(); + + @Foo({ + "a" + "\\nb", + "c" + "\\nd", + }) + A trailingComma3(); + } + + """) .addOutputLines( "out/pkg/A.java", - "package pkg;", - "", - "import pkg.A.Foo;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] value2() default {};", - " }", - "", - " @pkg.A.Foo", - " A functional1();", - "", - " @A.Foo", - " A functional2();", - "", - " @Foo", - " A functional3();", - "", - " @pkg.A.Foo(\"foo\")", - " A verbose1();", - "", - " @A.Foo(\"a'b\")", - " A verbose2();", - "", - " @Foo(\"a\" + \"\\nb\")", - " A verbose3();", - "", - " @pkg.A.Foo(\"foo\")", - " A moreVerbose1();", - "", - " @A.Foo(\"a'b\")", - " A moreVerbose2();", - "", - " @Foo(\"a\" + \"\\nb\")", - " A moreVerbose3();", - "", - " @pkg.A.Foo(", - " value = {\"foo\", \"bar\"},", - " value2 = 2)", - " A extended1();", - "", - " @A.Foo(", - " value = {\"a'b\", \"c'd\"},", - " value2 = 2)", - " A extended2();", - "", - " @Foo(", - " value = {\"a\" + \"\\nb\", \"c\" + \"\\nd\"},", - " value2 = 2)", - " A extended3();", - "", - " @pkg.A.Foo({\"foo\", \"bar\"})", - " A trailingComma1();", - "", - " @A.Foo({\"a'b\", \"c'd\"})", - " A trailingComma2();", - "", - " @Foo({\"a\" + \"\\nb\", \"c\" + \"\\nd\"})", - " A trailingComma3();", - "}") + """ + package pkg; + + import pkg.A.Foo; + + interface A { + @interface Foo { + String[] value() default {}; + + int[] value2() default {}; + } + + @pkg.A.Foo + A functional1(); + + @A.Foo + A functional2(); + + @Foo + A functional3(); + + @pkg.A.Foo("foo") + A verbose1(); + + @A.Foo("a'b") + A verbose2(); + + @Foo("a" + "\\nb") + A verbose3(); + + @pkg.A.Foo("foo") + A moreVerbose1(); + + @A.Foo("a'b") + A moreVerbose2(); + + @Foo("a" + "\\nb") + A moreVerbose3(); + + @pkg.A.Foo( + value = {"foo", "bar"}, + value2 = 2) + A extended1(); + + @A.Foo( + value = {"a'b", "c'd"}, + value2 = 2) + A extended2(); + + @Foo( + value = {"a" + "\\nb", "c" + "\\nd"}, + value2 = 2) + A extended3(); + + @pkg.A.Foo({"foo", "bar"}) + A trailingComma1(); + + @A.Foo({"a'b", "c'd"}) + A trailingComma2(); + + @Foo({"a" + "\\nb", "c" + "\\nd"}) + A trailingComma3(); + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CollectorMutabilityTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CollectorMutabilityTest.java index 48509379e5f..f7c72778e60 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CollectorMutabilityTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/CollectorMutabilityTest.java @@ -18,50 +18,53 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static com.google.common.collect.ImmutableList.toImmutableList;", - "import static com.google.common.collect.ImmutableMap.toImmutableMap;", - "import static com.google.common.collect.ImmutableSet.toImmutableSet;", - "import static java.util.stream.Collectors.toCollection;", - "import static java.util.stream.Collectors.toList;", - "import static java.util.stream.Collectors.toMap;", - "import static java.util.stream.Collectors.toSet;", - "", - "import java.util.ArrayList;", - "import java.util.HashMap;", - "import java.util.HashSet;", - "import java.util.stream.Collectors;", - "import java.util.stream.Stream;", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " // BUG: Diagnostic contains:", - " Flux.just(1).collect(Collectors.toList());", - " // BUG: Diagnostic contains:", - " Flux.just(2).collect(toList());", - " Flux.just(3).collect(toImmutableList());", - " Flux.just(4).collect(toCollection(ArrayList::new));", - "", - " // BUG: Diagnostic contains:", - " Flux.just(\"foo\").collect(Collectors.toMap(String::getBytes, String::length));", - " // BUG: Diagnostic contains:", - " Flux.just(\"bar\").collect(toMap(String::getBytes, String::length));", - " Flux.just(\"baz\").collect(toImmutableMap(String::getBytes, String::length));", - " // BUG: Diagnostic contains:", - " Flux.just(\"qux\").collect(toMap(String::getBytes, String::length, (a, b) -> a));", - " Flux.just(\"quux\").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> a));", - " Flux.just(\"quuz\").collect(toMap(String::getBytes, String::length, (a, b) -> a, HashMap::new));", - "", - " // BUG: Diagnostic contains:", - " Stream.of(1).collect(Collectors.toSet());", - " // BUG: Diagnostic contains:", - " Stream.of(2).collect(toSet());", - " Stream.of(3).collect(toImmutableSet());", - " Stream.of(4).collect(toCollection(HashSet::new));", - "", - " Flux.just(\"foo\").collect(Collectors.joining());", - " }", - "}") + """ + import static com.google.common.collect.ImmutableList.toImmutableList; + import static com.google.common.collect.ImmutableMap.toImmutableMap; + import static com.google.common.collect.ImmutableSet.toImmutableSet; + import static java.util.stream.Collectors.toCollection; + import static java.util.stream.Collectors.toList; + import static java.util.stream.Collectors.toMap; + import static java.util.stream.Collectors.toSet; + + import java.util.ArrayList; + import java.util.HashMap; + import java.util.HashSet; + import java.util.stream.Collectors; + import java.util.stream.Stream; + import reactor.core.publisher.Flux; + + class A { + void m() { + // BUG: Diagnostic contains: + Flux.just(1).collect(Collectors.toList()); + // BUG: Diagnostic contains: + Flux.just(2).collect(toList()); + Flux.just(3).collect(toImmutableList()); + Flux.just(4).collect(toCollection(ArrayList::new)); + + // BUG: Diagnostic contains: + Flux.just("foo").collect(Collectors.toMap(String::getBytes, String::length)); + // BUG: Diagnostic contains: + Flux.just("bar").collect(toMap(String::getBytes, String::length)); + Flux.just("baz").collect(toImmutableMap(String::getBytes, String::length)); + // BUG: Diagnostic contains: + Flux.just("qux").collect(toMap(String::getBytes, String::length, (a, b) -> a)); + Flux.just("quux").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> a)); + Flux.just("quuz").collect(toMap(String::getBytes, String::length, (a, b) -> a, HashMap::new)); + + // BUG: Diagnostic contains: + Stream.of(1).collect(Collectors.toSet()); + // BUG: Diagnostic contains: + Stream.of(2).collect(toSet()); + Stream.of(3).collect(toImmutableSet()); + Stream.of(4).collect(toCollection(HashSet::new)); + + Flux.just("foo").collect(Collectors.joining()); + } + } + + """) .doTest(); } @@ -70,55 +73,61 @@ void replacementFirstSuggestedFix() { refactoringTestHelper .addInputLines( "A.java", - "import static java.util.stream.Collectors.toList;", - "import static java.util.stream.Collectors.toMap;", - "import static java.util.stream.Collectors.toSet;", - "", - "import java.util.stream.Collectors;", - "import java.util.stream.Stream;", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).collect(Collectors.toList());", - " Flux.just(2).collect(toList());", - "", - " Stream.of(\"foo\").collect(Collectors.toMap(String::getBytes, String::length));", - " Stream.of(\"bar\").collect(toMap(String::getBytes, String::length));", - " Flux.just(\"baz\").collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b));", - " Flux.just(\"qux\").collect(toMap(String::getBytes, String::length, (a, b) -> b));", - "", - " Stream.of(1).collect(Collectors.toSet());", - " Stream.of(2).collect(toSet());", - " }", - "}") + """ + import static java.util.stream.Collectors.toList; + import static java.util.stream.Collectors.toMap; + import static java.util.stream.Collectors.toSet; + + import java.util.stream.Collectors; + import java.util.stream.Stream; + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).collect(Collectors.toList()); + Flux.just(2).collect(toList()); + + Stream.of("foo").collect(Collectors.toMap(String::getBytes, String::length)); + Stream.of("bar").collect(toMap(String::getBytes, String::length)); + Flux.just("baz").collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b)); + Flux.just("qux").collect(toMap(String::getBytes, String::length, (a, b) -> b)); + + Stream.of(1).collect(Collectors.toSet()); + Stream.of(2).collect(toSet()); + } + } + + """) .addOutputLines( "A.java", - "import static com.google.common.collect.ImmutableList.toImmutableList;", - "import static com.google.common.collect.ImmutableMap.toImmutableMap;", - "import static com.google.common.collect.ImmutableSet.toImmutableSet;", - "import static java.util.stream.Collectors.toList;", - "import static java.util.stream.Collectors.toMap;", - "import static java.util.stream.Collectors.toSet;", - "", - "import java.util.stream.Collectors;", - "import java.util.stream.Stream;", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).collect(toImmutableList());", - " Flux.just(2).collect(toImmutableList());", - "", - " Stream.of(\"foo\").collect(toImmutableMap(String::getBytes, String::length));", - " Stream.of(\"bar\").collect(toImmutableMap(String::getBytes, String::length));", - " Flux.just(\"baz\").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> b));", - " Flux.just(\"qux\").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> b));", - "", - " Stream.of(1).collect(toImmutableSet());", - " Stream.of(2).collect(toImmutableSet());", - " }", - "}") + """ + import static com.google.common.collect.ImmutableList.toImmutableList; + import static com.google.common.collect.ImmutableMap.toImmutableMap; + import static com.google.common.collect.ImmutableSet.toImmutableSet; + import static java.util.stream.Collectors.toList; + import static java.util.stream.Collectors.toMap; + import static java.util.stream.Collectors.toSet; + + import java.util.stream.Collectors; + import java.util.stream.Stream; + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).collect(toImmutableList()); + Flux.just(2).collect(toImmutableList()); + + Stream.of("foo").collect(toImmutableMap(String::getBytes, String::length)); + Stream.of("bar").collect(toImmutableMap(String::getBytes, String::length)); + Flux.just("baz").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> b)); + Flux.just("qux").collect(toImmutableMap(String::getBytes, String::length, (a, b) -> b)); + + Stream.of(1).collect(toImmutableSet()); + Stream.of(2).collect(toImmutableSet()); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -128,73 +137,79 @@ void replacementSecondSuggestedFix() { .setFixChooser(SECOND) .addInputLines( "A.java", - "import static java.util.stream.Collectors.toList;", - "import static java.util.stream.Collectors.toMap;", - "import static java.util.stream.Collectors.toSet;", - "", - "import java.util.stream.Collectors;", - "import java.util.stream.Stream;", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).collect(Collectors.toList());", - " Flux.just(2).collect(toList());", - "", - " Stream.of(\"foo\").collect(Collectors.toMap(String::getBytes, String::length));", - " Stream.of(\"bar\").collect(toMap(String::getBytes, String::length));", - " Flux.just(\"baz\").collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b));", - " Flux.just(\"qux\").collect(toMap(String::getBytes, String::length, (a, b) -> b));", - "", - " Stream.of(1).collect(Collectors.toSet());", - " Stream.of(2).collect(toSet());", - " }", - "}") + """ + import static java.util.stream.Collectors.toList; + import static java.util.stream.Collectors.toMap; + import static java.util.stream.Collectors.toSet; + + import java.util.stream.Collectors; + import java.util.stream.Stream; + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).collect(Collectors.toList()); + Flux.just(2).collect(toList()); + + Stream.of("foo").collect(Collectors.toMap(String::getBytes, String::length)); + Stream.of("bar").collect(toMap(String::getBytes, String::length)); + Flux.just("baz").collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b)); + Flux.just("qux").collect(toMap(String::getBytes, String::length, (a, b) -> b)); + + Stream.of(1).collect(Collectors.toSet()); + Stream.of(2).collect(toSet()); + } + } + + """) .addOutputLines( "A.java", - "import static java.util.stream.Collectors.toCollection;", - "import static java.util.stream.Collectors.toList;", - "import static java.util.stream.Collectors.toMap;", - "import static java.util.stream.Collectors.toSet;", - "", - "import java.util.ArrayList;", - "import java.util.HashMap;", - "import java.util.HashSet;", - "import java.util.stream.Collectors;", - "import java.util.stream.Stream;", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).collect(toCollection(ArrayList::new));", - " Flux.just(2).collect(toCollection(ArrayList::new));", - "", - " Stream.of(\"foo\")", - " .collect(", - " Collectors.toMap(", - " String::getBytes,", - " String::length,", - " (a, b) -> {", - " throw new IllegalStateException();", - " },", - " HashMap::new));", - " Stream.of(\"bar\")", - " .collect(", - " toMap(", - " String::getBytes,", - " String::length,", - " (a, b) -> {", - " throw new IllegalStateException();", - " },", - " HashMap::new));", - " Flux.just(\"baz\")", - " .collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b, HashMap::new));", - " Flux.just(\"qux\").collect(toMap(String::getBytes, String::length, (a, b) -> b, HashMap::new));", - "", - " Stream.of(1).collect(toCollection(HashSet::new));", - " Stream.of(2).collect(toCollection(HashSet::new));", - " }", - "}") + """ + import static java.util.stream.Collectors.toCollection; + import static java.util.stream.Collectors.toList; + import static java.util.stream.Collectors.toMap; + import static java.util.stream.Collectors.toSet; + + import java.util.ArrayList; + import java.util.HashMap; + import java.util.HashSet; + import java.util.stream.Collectors; + import java.util.stream.Stream; + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).collect(toCollection(ArrayList::new)); + Flux.just(2).collect(toCollection(ArrayList::new)); + + Stream.of("foo") + .collect( + Collectors.toMap( + String::getBytes, + String::length, + (a, b) -> { + throw new IllegalStateException(); + }, + HashMap::new)); + Stream.of("bar") + .collect( + toMap( + String::getBytes, + String::length, + (a, b) -> { + throw new IllegalStateException(); + }, + HashMap::new)); + Flux.just("baz") + .collect(Collectors.toMap(String::getBytes, String::length, (a, b) -> b, HashMap::new)); + Flux.just("qux").collect(toMap(String::getBytes, String::length, (a, b) -> b, HashMap::new)); + + Stream.of(1).collect(toCollection(HashSet::new)); + Stream.of(2).collect(toCollection(HashSet::new)); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EmptyMethodTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EmptyMethodTest.java index 51ad68f9175..63b5dcc4cab 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EmptyMethodTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/EmptyMethodTest.java @@ -16,54 +16,60 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "class A {", - " Object m1() {", - " return null;", - " }", - "", - " void m2() {", - " System.out.println(42);", - " }", - "", - " void m3() {}", - "", - " // BUG: Diagnostic contains:", - " static void m4() {}", - "", - " interface F {", - " void fun();", - " }", - "", - " final class MyTestClass {", - " void helperMethod() {}", - " }", - "}") + """ + class A { + Object m1() { + return null; + } + + void m2() { + System.out.println(42); + } + + void m3() {} + + // BUG: Diagnostic contains: + static void m4() {} + + interface F { + void fun(); + } + + final class MyTestClass { + void helperMethod() {} + } + } + + """) .addSourceLines( "B.java", - "import org.aspectj.lang.annotation.Pointcut;", - "", - "final class B implements A.F {", - " @Override", - " public void fun() {}", - "", - " // BUG: Diagnostic contains:", - " void m3() {}", - "", - " /** Javadoc. */", - " // BUG: Diagnostic contains:", - " void m4() {}", - "", - " void m5() {", - " // Single-line comment.", - " }", - "", - " void m6() {", - " /* Multi-line comment. */", - " }", - "", - " @Pointcut", - " void m7() {}", - "}") + """ + import org.aspectj.lang.annotation.Pointcut; + + final class B implements A.F { + @Override + public void fun() {} + + // BUG: Diagnostic contains: + void m3() {} + + /** Javadoc. */ + // BUG: Diagnostic contains: + void m4() {} + + void m5() { + // Single-line comment. + } + + void m6() { + /* Multi-line comment. */ + } + + @Pointcut + void m7() {} + } + + """) .doTest(); } @@ -72,12 +78,18 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "final class A {", - " void instanceMethod() {}", - "", - " static void staticMethod() {}", - "}") - .addOutputLines("out/A.java", "final class A {}") + """ + final class A { + void instanceMethod() {} + + static void staticMethod() {} + } + + """) + .addOutputLines("out/A.java", """ + final class A {} + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ErrorProneTestHelperSourceFormatTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ErrorProneTestHelperSourceFormatTest.java index 71a1f5b2bcf..9707f632b29 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ErrorProneTestHelperSourceFormatTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ErrorProneTestHelperSourceFormatTest.java @@ -22,63 +22,66 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.google.errorprone.BugCheckerRefactoringTestHelper;", - "import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode;", - "import com.google.errorprone.CompilationTestHelper;", - "import tech.picnic.errorprone.bugpatterns.EmptyMethod;", - "", - "class A {", - " private final CompilationTestHelper compilationTestHelper =", - " CompilationTestHelper.newInstance(EmptyMethod.class, getClass());", - " private final BugCheckerRefactoringTestHelper refactoringTestHelper =", - " BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass());", - "", - " void m() {", - " compilationTestHelper", - " // BUG: Diagnostic contains: No source code provided", - " .addSourceLines(\"A.java\")", - " // BUG: Diagnostic contains: Source code is malformed:", - " .addSourceLines(\"B.java\", \"class B {\")", - " // BUG: Diagnostic contains: Test code should be specified using a single text block", - " .addSourceLines(\"C.java\", \"class C {}\")", - " // Malformed code, but not compile-time constant, so not flagged.", - " .addSourceLines(\"D.java\", \"class D {\" + getClass())", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addSourceLines(\"E.java\", \"class E { }\")", - " // Well-formed code, so not flagged.", - " .addSourceLines(\"F.java\", \"\"\"", - " class F {}", - " \"\"\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style (pay attention to", - " // trailing newlines)", - " .addSourceLines(\"G.java\", \"\"\"", - " class G {}\"\"\")", - " .doTest();", - "", - " refactoringTestHelper", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addInputLines(\"in/A.java\", \"class A { }\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addOutputLines(\"out/A.java\", \"class A { }\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addInputLines(", - " \"in/B.java\",", - " \"\"\"", - " import java.util.Map;", - "", - " class B {}", - " \"\"\")", - " // Unused import, but in an output file, so not flagged.", - " .addOutputLines(", - " \"out/B.java\",", - " \"\"\"", - " import java.util.Map;", - "", - " class B {}", - " \"\"\")", - " .doTest(TestMode.TEXT_MATCH);", - " }", - "}") + """ + import com.google.errorprone.BugCheckerRefactoringTestHelper; + import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode; + import com.google.errorprone.CompilationTestHelper; + import tech.picnic.errorprone.bugpatterns.EmptyMethod; + + class A { + private final CompilationTestHelper compilationTestHelper = + CompilationTestHelper.newInstance(EmptyMethod.class, getClass()); + private final BugCheckerRefactoringTestHelper refactoringTestHelper = + BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass()); + + void m() { + compilationTestHelper + // BUG: Diagnostic contains: No source code provided + .addSourceLines("A.java") + // BUG: Diagnostic contains: Source code is malformed: + .addSourceLines("B.java", "class B {") + // BUG: Diagnostic contains: Test code should be specified using a single text block + .addSourceLines("C.java", "class C {}") + // Malformed code, but not compile-time constant, so not flagged. + .addSourceLines("D.java", "class D {" + getClass()) + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addSourceLines("E.java", "class E { }") + // Well-formed code, so not flagged. + .addSourceLines("F.java", ""\" + class F {} + ""\") + // BUG: Diagnostic contains: Test code should follow the Google Java style (pay attention to + // trailing newlines) + .addSourceLines("G.java", ""\" + class G {}""\") + .doTest(); + + refactoringTestHelper + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addInputLines("in/A.java", "class A { }") + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addOutputLines("out/A.java", "class A { }") + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addInputLines( + "in/B.java", + ""\" + import java.util.Map; + + class B {} + ""\") + // Unused import, but in an output file, so not flagged. + .addOutputLines( + "out/B.java", + ""\" + import java.util.Map; + + class B {} + ""\") + .doTest(TestMode.TEXT_MATCH); + } + } + + """) .doTest(); } @@ -87,50 +90,53 @@ void identificationAvoidTextBlocks() { avoidTextBlocksCompilationTestHelper .addSourceLines( "A.java", - "import com.google.errorprone.BugCheckerRefactoringTestHelper;", - "import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode;", - "import com.google.errorprone.CompilationTestHelper;", - "import tech.picnic.errorprone.bugpatterns.EmptyMethod;", - "", - "class A {", - " private final CompilationTestHelper compilationTestHelper =", - " CompilationTestHelper.newInstance(EmptyMethod.class, getClass());", - " private final BugCheckerRefactoringTestHelper refactoringTestHelper =", - " BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass());", - "", - " void m() {", - " compilationTestHelper", - " // BUG: Diagnostic contains: No source code provided", - " .addSourceLines(\"A.java\")", - " // BUG: Diagnostic contains: Source code is malformed:", - " .addSourceLines(\"B.java\", \"class B {\")", - " // Well-formed code, so not flagged.", - " .addSourceLines(\"C.java\", \"class C {}\")", - " // Malformed code, but not compile-time constant, so not flagged.", - " .addSourceLines(\"D.java\", \"class D {\" + getClass())", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addSourceLines(\"E.java\", \"class E { }\")", - " // BUG: Diagnostic contains: Test code should not be specified using a single text block", - " .addSourceLines(\"F.java\", \"\"\"", - " class F {}", - " \"\"\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style (pay attention to", - " // trailing newlines)", - " .addSourceLines(\"G.java\", \"class G {}\", \"\")", - " .doTest();", - "", - " refactoringTestHelper", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addInputLines(\"in/A.java\", \"class A { }\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addOutputLines(\"out/A.java\", \"class A { }\")", - " // BUG: Diagnostic contains: Test code should follow the Google Java style", - " .addInputLines(\"in/B.java\", \"import java.util.Map;\", \"\", \"class B {}\")", - " // Unused import, but in an output file, so not flagged.", - " .addOutputLines(\"out/B.java\", \"import java.util.Map;\", \"\", \"class B {}\")", - " .doTest(TestMode.TEXT_MATCH);", - " }", - "}") + """ + import com.google.errorprone.BugCheckerRefactoringTestHelper; + import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode; + import com.google.errorprone.CompilationTestHelper; + import tech.picnic.errorprone.bugpatterns.EmptyMethod; + + class A { + private final CompilationTestHelper compilationTestHelper = + CompilationTestHelper.newInstance(EmptyMethod.class, getClass()); + private final BugCheckerRefactoringTestHelper refactoringTestHelper = + BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass()); + + void m() { + compilationTestHelper + // BUG: Diagnostic contains: No source code provided + .addSourceLines("A.java") + // BUG: Diagnostic contains: Source code is malformed: + .addSourceLines("B.java", "class B {") + // Well-formed code, so not flagged. + .addSourceLines("C.java", "class C {}") + // Malformed code, but not compile-time constant, so not flagged. + .addSourceLines("D.java", "class D {" + getClass()) + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addSourceLines("E.java", "class E { }") + // BUG: Diagnostic contains: Test code should not be specified using a single text block + .addSourceLines("F.java", ""\" + class F {} + ""\") + // BUG: Diagnostic contains: Test code should follow the Google Java style (pay attention to + // trailing newlines) + .addSourceLines("G.java", "class G {}", "") + .doTest(); + + refactoringTestHelper + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addInputLines("in/A.java", "class A { }") + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addOutputLines("out/A.java", "class A { }") + // BUG: Diagnostic contains: Test code should follow the Google Java style + .addInputLines("in/B.java", "import java.util.Map;", "", "class B {}") + // Unused import, but in an output file, so not flagged. + .addOutputLines("out/B.java", "import java.util.Map;", "", "class B {}") + .doTest(TestMode.TEXT_MATCH); + } + } + + """) .doTest(); } @@ -145,86 +151,92 @@ void replacementAvoidTextBlocks() { avoidTextBlocksRefactoringTestHelper .addInputLines( "in/A.java", - "import com.google.errorprone.BugCheckerRefactoringTestHelper;", - "import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode;", - "import com.google.errorprone.CompilationTestHelper;", - "import tech.picnic.errorprone.bugpatterns.EmptyMethod;", - "", - "class A {", - " private final CompilationTestHelper compilationTestHelper =", - " CompilationTestHelper.newInstance(EmptyMethod.class, getClass());", - " private final BugCheckerRefactoringTestHelper refactoringTestHelper =", - " BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass());", - "", - " void m() {", - " compilationTestHelper", - " .addSourceLines(", - " \"A.java\",", - " \"import java.util.Map;\",", - " \"import java.util.Collection;\",", - " \"import java.util.List;\",", - " \"\",", - " \"interface A extends List, Map { }\")", - " .doTest();", - "", - " refactoringTestHelper", - " .addInputLines(", - " \"in/A.java\",", - " \"import java.util.Map;\",", - " \"import java.util.Collection;\",", - " \"import java.util.List;\",", - " \"\",", - " \"interface A extends List, Map { }\")", - " .addOutputLines(", - " \"out/A.java\",", - " \"import java.util.Map;\",", - " \"import java.util.Collection;\",", - " \"import java.util.List;\",", - " \"\",", - " \"interface A extends List, Map { }\")", - " .doTest(TestMode.TEXT_MATCH);", - " }", - "}") + """ + import com.google.errorprone.BugCheckerRefactoringTestHelper; + import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode; + import com.google.errorprone.CompilationTestHelper; + import tech.picnic.errorprone.bugpatterns.EmptyMethod; + + class A { + private final CompilationTestHelper compilationTestHelper = + CompilationTestHelper.newInstance(EmptyMethod.class, getClass()); + private final BugCheckerRefactoringTestHelper refactoringTestHelper = + BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass()); + + void m() { + compilationTestHelper + .addSourceLines( + "A.java", + "import java.util.Map;", + "import java.util.Collection;", + "import java.util.List;", + "", + "interface A extends List, Map { }") + .doTest(); + + refactoringTestHelper + .addInputLines( + "in/A.java", + "import java.util.Map;", + "import java.util.Collection;", + "import java.util.List;", + "", + "interface A extends List, Map { }") + .addOutputLines( + "out/A.java", + "import java.util.Map;", + "import java.util.Collection;", + "import java.util.List;", + "", + "interface A extends List, Map { }") + .doTest(TestMode.TEXT_MATCH); + } + } + + """) .addOutputLines( "out/A.java", - "import com.google.errorprone.BugCheckerRefactoringTestHelper;", - "import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode;", - "import com.google.errorprone.CompilationTestHelper;", - "import tech.picnic.errorprone.bugpatterns.EmptyMethod;", - "", - "class A {", - " private final CompilationTestHelper compilationTestHelper =", - " CompilationTestHelper.newInstance(EmptyMethod.class, getClass());", - " private final BugCheckerRefactoringTestHelper refactoringTestHelper =", - " BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass());", - "", - " void m() {", - " compilationTestHelper", - " .addSourceLines(", - " \"A.java\",", - " \"import java.util.List;\",", - " \"import java.util.Map;\",", - " \"\",", - " \"interface A extends List, Map {}\")", - " .doTest();", - "", - " refactoringTestHelper", - " .addInputLines(", - " \"in/A.java\",", - " \"import java.util.List;\",", - " \"import java.util.Map;\",", - " \"\",", - " \"interface A extends List, Map {}\")", - " .addOutputLines(", - " \"out/A.java\",", - " \"import java.util.Collection;\",", - " \"import java.util.List;\",", - " \"import java.util.Map;\",", - " \"\",", - " \"interface A extends List, Map {}\")", - " .doTest(TestMode.TEXT_MATCH);", - " }", - "}") + """ + import com.google.errorprone.BugCheckerRefactoringTestHelper; + import com.google.errorprone.BugCheckerRefactoringTestHelper.TestMode; + import com.google.errorprone.CompilationTestHelper; + import tech.picnic.errorprone.bugpatterns.EmptyMethod; + + class A { + private final CompilationTestHelper compilationTestHelper = + CompilationTestHelper.newInstance(EmptyMethod.class, getClass()); + private final BugCheckerRefactoringTestHelper refactoringTestHelper = + BugCheckerRefactoringTestHelper.newInstance(EmptyMethod.class, getClass()); + + void m() { + compilationTestHelper + .addSourceLines( + "A.java", + "import java.util.List;", + "import java.util.Map;", + "", + "interface A extends List, Map {}") + .doTest(); + + refactoringTestHelper + .addInputLines( + "in/A.java", + "import java.util.List;", + "import java.util.Map;", + "", + "interface A extends List, Map {}") + .addOutputLines( + "out/A.java", + "import java.util.Collection;", + "import java.util.List;", + "import java.util.Map;", + "", + "interface A extends List, Map {}") + .doTest(TestMode.TEXT_MATCH); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ExplicitEnumOrderingTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ExplicitEnumOrderingTest.java index 3a8119a0ce1..5dd1d912e32 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ExplicitEnumOrderingTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ExplicitEnumOrderingTest.java @@ -12,74 +12,77 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static java.lang.annotation.RetentionPolicy.CLASS;", - "import static java.lang.annotation.RetentionPolicy.RUNTIME;", - "import static java.lang.annotation.RetentionPolicy.SOURCE;", - "import static java.time.chrono.IsoEra.BCE;", - "import static java.time.chrono.IsoEra.CE;", - "", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.Ordering;", - "import java.lang.annotation.RetentionPolicy;", - "import java.time.chrono.IsoEra;", - "", - "class A {", - " {", - " // The `List`-accepting overload is currently ignored.", - " Ordering.explicit(ImmutableList.of(RetentionPolicy.SOURCE, RetentionPolicy.CLASS));", - "", - " Ordering.explicit(IsoEra.BCE, IsoEra.CE);", - " // BUG: Diagnostic contains: IsoEra.CE", - " Ordering.explicit(IsoEra.BCE);", - " // BUG: Diagnostic contains: IsoEra.BCE", - " Ordering.explicit(IsoEra.CE);", - "", - " Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.CLASS, RetentionPolicy.RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS, RetentionPolicy.RUNTIME", - " Ordering.explicit(RetentionPolicy.SOURCE);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME", - " Ordering.explicit(RetentionPolicy.CLASS);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.CLASS", - " Ordering.explicit(RetentionPolicy.RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.RUNTIME", - " Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.CLASS);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS", - " Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE", - " Ordering.explicit(RetentionPolicy.CLASS, RetentionPolicy.RUNTIME);", - "", - " Ordering.explicit(BCE, CE);", - " // BUG: Diagnostic contains: IsoEra.CE", - " Ordering.explicit(BCE);", - " // BUG: Diagnostic contains: IsoEra.BCE", - " Ordering.explicit(CE);", - "", - " Ordering.explicit(SOURCE, CLASS, RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS, RetentionPolicy.RUNTIME", - " Ordering.explicit(SOURCE);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME", - " Ordering.explicit(CLASS);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.CLASS", - " Ordering.explicit(RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.RUNTIME", - " Ordering.explicit(SOURCE, CLASS);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS", - " Ordering.explicit(SOURCE, RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE", - " Ordering.explicit(CLASS, RUNTIME);", - "", - " Ordering.explicit(RetentionPolicy.SOURCE, BCE, RetentionPolicy.CLASS, CE, RUNTIME);", - " Ordering.explicit(SOURCE, IsoEra.BCE, CLASS, IsoEra.CE, RetentionPolicy.RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS", - " Ordering.explicit(RetentionPolicy.SOURCE, BCE, CE, RUNTIME);", - " // BUG: Diagnostic contains: RetentionPolicy.CLASS", - " Ordering.explicit(IsoEra.BCE, SOURCE, IsoEra.CE, RetentionPolicy.RUNTIME);", - " // BUG: Diagnostic contains: IsoEra.CE, RetentionPolicy.RUNTIME", - " Ordering.explicit(IsoEra.BCE, SOURCE, RetentionPolicy.CLASS);", - " // BUG: Diagnostic contains: RetentionPolicy.SOURCE, IsoEra.BCE", - " Ordering.explicit(CLASS, RUNTIME, CE);", - " }", - "}") + """ + import static java.lang.annotation.RetentionPolicy.CLASS; + import static java.lang.annotation.RetentionPolicy.RUNTIME; + import static java.lang.annotation.RetentionPolicy.SOURCE; + import static java.time.chrono.IsoEra.BCE; + import static java.time.chrono.IsoEra.CE; + + import com.google.common.collect.ImmutableList; + import com.google.common.collect.Ordering; + import java.lang.annotation.RetentionPolicy; + import java.time.chrono.IsoEra; + + class A { + { + // The `List`-accepting overload is currently ignored. + Ordering.explicit(ImmutableList.of(RetentionPolicy.SOURCE, RetentionPolicy.CLASS)); + + Ordering.explicit(IsoEra.BCE, IsoEra.CE); + // BUG: Diagnostic contains: IsoEra.CE + Ordering.explicit(IsoEra.BCE); + // BUG: Diagnostic contains: IsoEra.BCE + Ordering.explicit(IsoEra.CE); + + Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.CLASS, RetentionPolicy.RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.CLASS, RetentionPolicy.RUNTIME + Ordering.explicit(RetentionPolicy.SOURCE); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME + Ordering.explicit(RetentionPolicy.CLASS); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.CLASS + Ordering.explicit(RetentionPolicy.RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.RUNTIME + Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.CLASS); + // BUG: Diagnostic contains: RetentionPolicy.CLASS + Ordering.explicit(RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE + Ordering.explicit(RetentionPolicy.CLASS, RetentionPolicy.RUNTIME); + + Ordering.explicit(BCE, CE); + // BUG: Diagnostic contains: IsoEra.CE + Ordering.explicit(BCE); + // BUG: Diagnostic contains: IsoEra.BCE + Ordering.explicit(CE); + + Ordering.explicit(SOURCE, CLASS, RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.CLASS, RetentionPolicy.RUNTIME + Ordering.explicit(SOURCE); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.RUNTIME + Ordering.explicit(CLASS); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE, RetentionPolicy.CLASS + Ordering.explicit(RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.RUNTIME + Ordering.explicit(SOURCE, CLASS); + // BUG: Diagnostic contains: RetentionPolicy.CLASS + Ordering.explicit(SOURCE, RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE + Ordering.explicit(CLASS, RUNTIME); + + Ordering.explicit(RetentionPolicy.SOURCE, BCE, RetentionPolicy.CLASS, CE, RUNTIME); + Ordering.explicit(SOURCE, IsoEra.BCE, CLASS, IsoEra.CE, RetentionPolicy.RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.CLASS + Ordering.explicit(RetentionPolicy.SOURCE, BCE, CE, RUNTIME); + // BUG: Diagnostic contains: RetentionPolicy.CLASS + Ordering.explicit(IsoEra.BCE, SOURCE, IsoEra.CE, RetentionPolicy.RUNTIME); + // BUG: Diagnostic contains: IsoEra.CE, RetentionPolicy.RUNTIME + Ordering.explicit(IsoEra.BCE, SOURCE, RetentionPolicy.CLASS); + // BUG: Diagnostic contains: RetentionPolicy.SOURCE, IsoEra.BCE + Ordering.explicit(CLASS, RUNTIME, CE); + } + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FluxFlatMapUsageTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FluxFlatMapUsageTest.java index a120fb02639..6919d156c48 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FluxFlatMapUsageTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FluxFlatMapUsageTest.java @@ -18,47 +18,50 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.function.BiFunction;", - "import java.util.function.Function;", - "import reactor.core.publisher.Flux;", - "import reactor.core.publisher.Mono;", - "", - "class A {", - " void m() {", - " // BUG: Diagnostic contains:", - " Flux.just(1).flatMap(Flux::just);", - " // BUG: Diagnostic contains:", - " Flux.just(1).flatMap(i -> Flux.just(String.valueOf(i)));", - " // BUG: Diagnostic contains:", - " Flux.just(1).flatMapSequential(Flux::just);", - " // BUG: Diagnostic contains:", - " Flux.just(1).flatMapSequential(i -> Flux.just(String.valueOf(i)));", - "", - " Mono.just(1).flatMap(Mono::just);", - " Flux.just(1).concatMap(Flux::just);", - "", - " Flux.just(1).flatMap(Flux::just, 1);", - " Flux.just(1).flatMap(Flux::just, 1, 1);", - " Flux.just(1).flatMap(Flux::just, throwable -> Flux.empty(), Flux::empty);", - "", - " Flux.just(1).flatMapSequential(Flux::just, 1);", - " Flux.just(1).flatMapSequential(Flux::just, 1, 1);", - "", - " // BUG: Diagnostic contains:", - " this.>sink(Flux::flatMap);", - " // BUG: Diagnostic contains:", - " this.>sink(Flux::flatMap);", - "", - " // BUG: Diagnostic contains:", - " this.>sink(Flux::flatMapSequential);", - " // BUG: Diagnostic contains:", - " this.>sink(Flux::flatMapSequential);", - "", - " this.>sink(Mono::flatMap);", - " }", - "", - " private void sink(BiFunction, P> fun) {}", - "}") + """ + import java.util.function.BiFunction; + import java.util.function.Function; + import reactor.core.publisher.Flux; + import reactor.core.publisher.Mono; + + class A { + void m() { + // BUG: Diagnostic contains: + Flux.just(1).flatMap(Flux::just); + // BUG: Diagnostic contains: + Flux.just(1).flatMap(i -> Flux.just(String.valueOf(i))); + // BUG: Diagnostic contains: + Flux.just(1).flatMapSequential(Flux::just); + // BUG: Diagnostic contains: + Flux.just(1).flatMapSequential(i -> Flux.just(String.valueOf(i))); + + Mono.just(1).flatMap(Mono::just); + Flux.just(1).concatMap(Flux::just); + + Flux.just(1).flatMap(Flux::just, 1); + Flux.just(1).flatMap(Flux::just, 1, 1); + Flux.just(1).flatMap(Flux::just, throwable -> Flux.empty(), Flux::empty); + + Flux.just(1).flatMapSequential(Flux::just, 1); + Flux.just(1).flatMapSequential(Flux::just, 1, 1); + + // BUG: Diagnostic contains: + this.>sink(Flux::flatMap); + // BUG: Diagnostic contains: + this.>sink(Flux::flatMap); + + // BUG: Diagnostic contains: + this.>sink(Flux::flatMapSequential); + // BUG: Diagnostic contains: + this.>sink(Flux::flatMapSequential); + + this.>sink(Mono::flatMap); + } + + private void sink(BiFunction, P> fun) {} + } + + """) .doTest(); } @@ -68,24 +71,30 @@ void replacementFirstSuggestedFix() { .setFixChooser(FixChoosers.FIRST) .addInputLines( "in/A.java", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).flatMap(Flux::just);", - " Flux.just(1).flatMapSequential(Flux::just);", - " }", - "}") + """ + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).flatMap(Flux::just); + Flux.just(1).flatMapSequential(Flux::just); + } + } + + """) .addOutputLines( "out/A.java", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " void m() {", - " Flux.just(1).concatMap(Flux::just);", - " Flux.just(1).concatMap(Flux::just);", - " }", - "}") + """ + import reactor.core.publisher.Flux; + + class A { + void m() { + Flux.just(1).concatMap(Flux::just); + Flux.just(1).concatMap(Flux::just); + } + } + + """) .doTest(); } @@ -95,28 +104,34 @@ void replacementSecondSuggestedFix() { .setFixChooser(FixChoosers.SECOND) .addInputLines( "in/A.java", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " private static final int MAX_CONCURRENCY = 8;", - "", - " void m() {", - " Flux.just(1).flatMap(Flux::just);", - " Flux.just(1).flatMapSequential(Flux::just);", - " }", - "}") + """ + import reactor.core.publisher.Flux; + + class A { + private static final int MAX_CONCURRENCY = 8; + + void m() { + Flux.just(1).flatMap(Flux::just); + Flux.just(1).flatMapSequential(Flux::just); + } + } + + """) .addOutputLines( "out/A.java", - "import reactor.core.publisher.Flux;", - "", - "class A {", - " private static final int MAX_CONCURRENCY = 8;", - "", - " void m() {", - " Flux.just(1).flatMap(Flux::just, MAX_CONCURRENCY);", - " Flux.just(1).flatMapSequential(Flux::just, MAX_CONCURRENCY);", - " }", - "}") + """ + import reactor.core.publisher.Flux; + + class A { + private static final int MAX_CONCURRENCY = 8; + + void m() { + Flux.just(1).flatMap(Flux::just, MAX_CONCURRENCY); + Flux.just(1).flatMapSequential(Flux::just, MAX_CONCURRENCY); + } + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FormatStringConcatenationTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FormatStringConcatenationTest.java index 737f00cc998..962739c749c 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FormatStringConcatenationTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/FormatStringConcatenationTest.java @@ -16,294 +16,297 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static com.google.common.base.Preconditions.checkArgument;", - "import static com.google.common.base.Preconditions.checkNotNull;", - "import static com.google.common.base.Preconditions.checkState;", - "import static com.google.common.base.Verify.verify;", - "import static org.assertj.core.api.Assertions.assertThat;", - "import static org.assertj.core.api.SoftAssertions.assertSoftly;", - "", - "import java.util.Formatter;", - "import java.util.Locale;", - "import org.assertj.core.api.Assertions;", - "import org.assertj.core.api.BDDAssertions;", - "import org.assertj.core.api.Fail;", - "import org.assertj.core.api.ThrowableAssertAlternative;", - "import org.assertj.core.api.WithAssertions;", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "", - "class A {", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " void negative() {", - " hashCode();", - " equals(new A());", - " equals(toString());", - " equals(0);", - " equals(\"str\");", - " equals(\"str\" + 0);", - " equals(0 + 0);", - " equals(0 - 0);", - " equals(\"str \" + toString());", - " }", - "", - " void assertj() {", - " assertThat(0).overridingErrorMessage(toString());", - " assertThat(0).overridingErrorMessage(\"str\");", - " assertThat(0).overridingErrorMessage(\"str \" + 0);", - " assertThat(0).overridingErrorMessage(\"str %s\", 2 * 3);", - " assertThat(0).overridingErrorMessage(\"str %s\", toString());", - " // BUG: Diagnostic contains:", - " assertThat(0).overridingErrorMessage(\"str \" + hashCode() / 2);", - " // BUG: Diagnostic contains:", - " assertThat(0).overridingErrorMessage((\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " assertThat(0).overridingErrorMessage(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(0).overridingErrorMessage(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(0).withFailMessage(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(0).withFailMessage(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertSoftly(softly -> softly.fail(\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " assertSoftly(softly -> softly.fail(\"%s \" + toString(), \"arg\"));", - " assertSoftly(softly -> softly.fail(\"str \" + toString(), new Throwable()));", - "", - " // BUG: Diagnostic contains:", - " assertThat(\"\").isEqualTo(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(\"\").isEqualTo(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessage(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessage(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageContaining(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageContaining(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageEndingWith(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageEndingWith(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageStartingWith(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasMessageStartingWith(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasRootCauseMessage(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasRootCauseMessage(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasStackTraceContaining(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(new Error()).hasStackTraceContaining(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(0).as(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(0).as(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " assertThat(0).describedAs(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " assertThat(0).describedAs(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessage(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessage(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageContaining(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageContaining(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageEndingWith(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageEndingWith(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageStartingWith(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withMessageStartingWith(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withStackTraceContaining(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((ThrowableAssertAlternative) null).withStackTraceContaining(\"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " ((WithAssertions) null).fail(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " ((WithAssertions) null).fail(\"%s \" + toString(), \"arg\");", - " ((WithAssertions) null).fail(\"str \" + toString(), new Throwable());", - "", - " // BUG: Diagnostic contains:", - " Assertions.fail(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " Assertions.fail(\"%s \" + toString(), \"arg\");", - " Assertions.fail(\"str \" + toString(), new Throwable());", - "", - " // BUG: Diagnostic contains:", - " BDDAssertions.fail(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " BDDAssertions.fail(\"%s \" + toString(), \"arg\");", - " BDDAssertions.fail(\"str \" + toString(), new Throwable());", - "", - " // BUG: Diagnostic contains:", - " Fail.fail(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " Fail.fail(\"%s \" + toString(), \"arg\");", - " Fail.fail(\"str \" + toString(), new Throwable());", - " }", - "", - " void guava() {", - " checkArgument(true);", - " checkArgument(true, toString());", - " checkArgument(true, \"str\");", - " checkArgument(true, \"str \" + 0);", - " checkArgument(true, \"str %s\", 2 * 3);", - " checkArgument(true, \"str %s\", toString());", - " // BUG: Diagnostic contains:", - " checkArgument(true, \"str \" + hashCode() / 2);", - " // BUG: Diagnostic contains:", - " checkArgument(true, (\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " checkArgument(true, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " checkArgument(true, \"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " checkNotNull(true, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " checkNotNull(true, \"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " checkState(true, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " checkState(true, \"%s \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " verify(true, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " verify(true, \"%s \" + toString(), \"arg\");", - " }", - "", - " void jdk() {", - " String.format(\"str\");", - " String.format(\"str \" + 0);", - " String.format(\"str {}\", 2 * 3);", - " String.format(\"str {}\", toString());", - " // BUG: Diagnostic contains:", - " String.format(\"str \" + hashCode() / 2);", - " // BUG: Diagnostic contains:", - " String.format((\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " String.format(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " String.format(\"{} \" + toString(), \"arg\");", - "", - " String.format(Locale.ROOT, \"str\");", - " String.format(Locale.ROOT, \"str \" + 0);", - " String.format(Locale.ROOT, \"str {}\", 2 * 3);", - " String.format(Locale.ROOT, \"str {}\", toString());", - " // BUG: Diagnostic contains:", - " String.format(Locale.ROOT, (\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " String.format(Locale.ROOT, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " String.format(Locale.ROOT, \"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " new Formatter().format(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " new Formatter().format(\"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " new Formatter().format(Locale.ROOT, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " new Formatter().format(Locale.ROOT, \"{} \" + toString(), \"arg\");", - " }", - "", - " void slf4j() {", - " LOG.debug(\"str\");", - " LOG.debug(\"str \" + 0);", - " LOG.debug(\"str {}\", 2 * 3);", - " LOG.debug(\"str {}\", toString());", - " // BUG: Diagnostic contains:", - " LOG.debug(\"str \" + hashCode() / 2);", - " // BUG: Diagnostic contains:", - " LOG.debug((\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " LOG.debug(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.debug(\"{} \" + toString(), \"arg\");", - "", - " LOG.debug((Marker) null, \"str\");", - " LOG.debug((Marker) null, \"str \" + 0);", - " LOG.debug((Marker) null, \"str {}\", 2 * 3);", - " LOG.debug((Marker) null, \"str {}\", toString());", - " // BUG: Diagnostic contains:", - " LOG.debug((Marker) null, (\"str \" + toString()));", - " // BUG: Diagnostic contains:", - " LOG.debug((Marker) null, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.debug((Marker) null, \"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.error(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.error(\"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.error((Marker) null, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.error((Marker) null, \"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.info(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.info(\"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.info((Marker) null, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.info((Marker) null, \"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.trace(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.trace(\"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.trace((Marker) null, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.trace((Marker) null, \"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.warn(\"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.warn(\"{} \" + toString(), \"arg\");", - "", - " // BUG: Diagnostic contains:", - " LOG.warn((Marker) null, \"str \" + toString());", - " // BUG: Diagnostic contains:", - " LOG.warn((Marker) null, \"{} \" + toString(), \"arg\");", - " }", - "}") + """ + import static com.google.common.base.Preconditions.checkArgument; + import static com.google.common.base.Preconditions.checkNotNull; + import static com.google.common.base.Preconditions.checkState; + import static com.google.common.base.Verify.verify; + import static org.assertj.core.api.Assertions.assertThat; + import static org.assertj.core.api.SoftAssertions.assertSoftly; + + import java.util.Formatter; + import java.util.Locale; + import org.assertj.core.api.Assertions; + import org.assertj.core.api.BDDAssertions; + import org.assertj.core.api.Fail; + import org.assertj.core.api.ThrowableAssertAlternative; + import org.assertj.core.api.WithAssertions; + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + + class A { + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + void negative() { + hashCode(); + equals(new A()); + equals(toString()); + equals(0); + equals("str"); + equals("str" + 0); + equals(0 + 0); + equals(0 - 0); + equals("str " + toString()); + } + + void assertj() { + assertThat(0).overridingErrorMessage(toString()); + assertThat(0).overridingErrorMessage("str"); + assertThat(0).overridingErrorMessage("str " + 0); + assertThat(0).overridingErrorMessage("str %s", 2 * 3); + assertThat(0).overridingErrorMessage("str %s", toString()); + // BUG: Diagnostic contains: + assertThat(0).overridingErrorMessage("str " + hashCode() / 2); + // BUG: Diagnostic contains: + assertThat(0).overridingErrorMessage(("str " + toString())); + // BUG: Diagnostic contains: + assertThat(0).overridingErrorMessage("str " + toString()); + // BUG: Diagnostic contains: + assertThat(0).overridingErrorMessage("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(0).withFailMessage("str " + toString()); + // BUG: Diagnostic contains: + assertThat(0).withFailMessage("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertSoftly(softly -> softly.fail("str " + toString())); + // BUG: Diagnostic contains: + assertSoftly(softly -> softly.fail("%s " + toString(), "arg")); + assertSoftly(softly -> softly.fail("str " + toString(), new Throwable())); + + // BUG: Diagnostic contains: + assertThat("").isEqualTo("str " + toString()); + // BUG: Diagnostic contains: + assertThat("").isEqualTo("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessage("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessage("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageContaining("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageContaining("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageEndingWith("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageEndingWith("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageStartingWith("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasMessageStartingWith("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasRootCauseMessage("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasRootCauseMessage("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(new Error()).hasStackTraceContaining("str " + toString()); + // BUG: Diagnostic contains: + assertThat(new Error()).hasStackTraceContaining("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(0).as("str " + toString()); + // BUG: Diagnostic contains: + assertThat(0).as("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + assertThat(0).describedAs("str " + toString()); + // BUG: Diagnostic contains: + assertThat(0).describedAs("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessage("str " + toString()); + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessage("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageContaining("str " + toString()); + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageContaining("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageEndingWith("str " + toString()); + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageEndingWith("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageStartingWith("str " + toString()); + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withMessageStartingWith("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withStackTraceContaining("str " + toString()); + // BUG: Diagnostic contains: + ((ThrowableAssertAlternative) null).withStackTraceContaining("%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + ((WithAssertions) null).fail("str " + toString()); + // BUG: Diagnostic contains: + ((WithAssertions) null).fail("%s " + toString(), "arg"); + ((WithAssertions) null).fail("str " + toString(), new Throwable()); + + // BUG: Diagnostic contains: + Assertions.fail("str " + toString()); + // BUG: Diagnostic contains: + Assertions.fail("%s " + toString(), "arg"); + Assertions.fail("str " + toString(), new Throwable()); + + // BUG: Diagnostic contains: + BDDAssertions.fail("str " + toString()); + // BUG: Diagnostic contains: + BDDAssertions.fail("%s " + toString(), "arg"); + BDDAssertions.fail("str " + toString(), new Throwable()); + + // BUG: Diagnostic contains: + Fail.fail("str " + toString()); + // BUG: Diagnostic contains: + Fail.fail("%s " + toString(), "arg"); + Fail.fail("str " + toString(), new Throwable()); + } + + void guava() { + checkArgument(true); + checkArgument(true, toString()); + checkArgument(true, "str"); + checkArgument(true, "str " + 0); + checkArgument(true, "str %s", 2 * 3); + checkArgument(true, "str %s", toString()); + // BUG: Diagnostic contains: + checkArgument(true, "str " + hashCode() / 2); + // BUG: Diagnostic contains: + checkArgument(true, ("str " + toString())); + // BUG: Diagnostic contains: + checkArgument(true, "str " + toString()); + // BUG: Diagnostic contains: + checkArgument(true, "%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + checkNotNull(true, "str " + toString()); + // BUG: Diagnostic contains: + checkNotNull(true, "%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + checkState(true, "str " + toString()); + // BUG: Diagnostic contains: + checkState(true, "%s " + toString(), "arg"); + + // BUG: Diagnostic contains: + verify(true, "str " + toString()); + // BUG: Diagnostic contains: + verify(true, "%s " + toString(), "arg"); + } + + void jdk() { + String.format("str"); + String.format("str " + 0); + String.format("str {}", 2 * 3); + String.format("str {}", toString()); + // BUG: Diagnostic contains: + String.format("str " + hashCode() / 2); + // BUG: Diagnostic contains: + String.format(("str " + toString())); + // BUG: Diagnostic contains: + String.format("str " + toString()); + // BUG: Diagnostic contains: + String.format("{} " + toString(), "arg"); + + String.format(Locale.ROOT, "str"); + String.format(Locale.ROOT, "str " + 0); + String.format(Locale.ROOT, "str {}", 2 * 3); + String.format(Locale.ROOT, "str {}", toString()); + // BUG: Diagnostic contains: + String.format(Locale.ROOT, ("str " + toString())); + // BUG: Diagnostic contains: + String.format(Locale.ROOT, "str " + toString()); + // BUG: Diagnostic contains: + String.format(Locale.ROOT, "{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + new Formatter().format("str " + toString()); + // BUG: Diagnostic contains: + new Formatter().format("{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + new Formatter().format(Locale.ROOT, "str " + toString()); + // BUG: Diagnostic contains: + new Formatter().format(Locale.ROOT, "{} " + toString(), "arg"); + } + + void slf4j() { + LOG.debug("str"); + LOG.debug("str " + 0); + LOG.debug("str {}", 2 * 3); + LOG.debug("str {}", toString()); + // BUG: Diagnostic contains: + LOG.debug("str " + hashCode() / 2); + // BUG: Diagnostic contains: + LOG.debug(("str " + toString())); + // BUG: Diagnostic contains: + LOG.debug("str " + toString()); + // BUG: Diagnostic contains: + LOG.debug("{} " + toString(), "arg"); + + LOG.debug((Marker) null, "str"); + LOG.debug((Marker) null, "str " + 0); + LOG.debug((Marker) null, "str {}", 2 * 3); + LOG.debug((Marker) null, "str {}", toString()); + // BUG: Diagnostic contains: + LOG.debug((Marker) null, ("str " + toString())); + // BUG: Diagnostic contains: + LOG.debug((Marker) null, "str " + toString()); + // BUG: Diagnostic contains: + LOG.debug((Marker) null, "{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.error("str " + toString()); + // BUG: Diagnostic contains: + LOG.error("{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.error((Marker) null, "str " + toString()); + // BUG: Diagnostic contains: + LOG.error((Marker) null, "{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.info("str " + toString()); + // BUG: Diagnostic contains: + LOG.info("{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.info((Marker) null, "str " + toString()); + // BUG: Diagnostic contains: + LOG.info((Marker) null, "{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.trace("str " + toString()); + // BUG: Diagnostic contains: + LOG.trace("{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.trace((Marker) null, "str " + toString()); + // BUG: Diagnostic contains: + LOG.trace((Marker) null, "{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.warn("str " + toString()); + // BUG: Diagnostic contains: + LOG.warn("{} " + toString(), "arg"); + + // BUG: Diagnostic contains: + LOG.warn((Marker) null, "str " + toString()); + // BUG: Diagnostic contains: + LOG.warn((Marker) null, "{} " + toString(), "arg"); + } + } + + """) .doTest(); } @@ -312,102 +315,108 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static com.google.common.base.Preconditions.checkArgument;", - "import static org.assertj.core.api.Assertions.assertThat;", - "", - "import java.util.Locale;", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "", - "class A {", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " void assertj() {", - " assertThat(0).overridingErrorMessage(toString() + \" str\");", - " assertThat(0).overridingErrorMessage(\"str \" + toString());", - " assertThat(0).overridingErrorMessage(toString() + toString());", - " assertThat(0).overridingErrorMessage(\"str \" + toString() + \" word \" + new A().hashCode());", - " assertThat(0).overridingErrorMessage(\"str \" + (toString() + \" word \") + (hashCode() / 2));", - "", - " // Flagged but not auto-fixed.", - " assertThat(0).overridingErrorMessage(\"%s \" + toString(), \"arg\");", - " }", - "", - " void guava() {", - " checkArgument(true, \"str \" + toString());", - "", - " // Flagged but not auto-fixed.", - " checkArgument(true, \"%s \" + toString(), \"arg\");", - " }", - "", - " void jdk() {", - " String.format(\"str \" + toString());", - " String.format(Locale.ROOT, \"str \" + toString());", - "", - " // Flagged but not auto-fixed.", - " String.format(\"{} \" + toString(), \"arg\");", - " String.format(Locale.ROOT, \"{} \" + toString(), \"arg\");", - " }", - "", - " void slf4j() {", - " LOG.debug(\"str \" + toString());", - " LOG.debug((Marker) null, \"str \" + toString());", - "", - " // Flagged but not auto-fixed.", - " LOG.debug(\"{} \" + toString(), \"arg\");", - " LOG.debug((Marker) null, \"{} \" + toString(), \"arg\");", - " }", - "}") + """ + import static com.google.common.base.Preconditions.checkArgument; + import static org.assertj.core.api.Assertions.assertThat; + + import java.util.Locale; + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + + class A { + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + void assertj() { + assertThat(0).overridingErrorMessage(toString() + " str"); + assertThat(0).overridingErrorMessage("str " + toString()); + assertThat(0).overridingErrorMessage(toString() + toString()); + assertThat(0).overridingErrorMessage("str " + toString() + " word " + new A().hashCode()); + assertThat(0).overridingErrorMessage("str " + (toString() + " word ") + (hashCode() / 2)); + + // Flagged but not auto-fixed. + assertThat(0).overridingErrorMessage("%s " + toString(), "arg"); + } + + void guava() { + checkArgument(true, "str " + toString()); + + // Flagged but not auto-fixed. + checkArgument(true, "%s " + toString(), "arg"); + } + + void jdk() { + String.format("str " + toString()); + String.format(Locale.ROOT, "str " + toString()); + + // Flagged but not auto-fixed. + String.format("{} " + toString(), "arg"); + String.format(Locale.ROOT, "{} " + toString(), "arg"); + } + + void slf4j() { + LOG.debug("str " + toString()); + LOG.debug((Marker) null, "str " + toString()); + + // Flagged but not auto-fixed. + LOG.debug("{} " + toString(), "arg"); + LOG.debug((Marker) null, "{} " + toString(), "arg"); + } + } + + """) .addOutputLines( "out/A.java", - "import static com.google.common.base.Preconditions.checkArgument;", - "import static org.assertj.core.api.Assertions.assertThat;", - "", - "import java.util.Locale;", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "", - "class A {", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " void assertj() {", - " assertThat(0).overridingErrorMessage(\"%s str\", toString());", - " assertThat(0).overridingErrorMessage(\"str %s\", toString());", - " assertThat(0).overridingErrorMessage(\"%s%s\", toString(), toString());", - " assertThat(0).overridingErrorMessage(\"str %s word %s\", toString(), new A().hashCode());", - " assertThat(0).overridingErrorMessage(\"str %s word %s\", toString(), hashCode() / 2);", - "", - " // Flagged but not auto-fixed.", - " assertThat(0).overridingErrorMessage(\"%s \" + toString(), \"arg\");", - " }", - "", - " void guava() {", - " checkArgument(true, \"str %s\", toString());", - "", - " // Flagged but not auto-fixed.", - " checkArgument(true, \"%s \" + toString(), \"arg\");", - " }", - "", - " void jdk() {", - " String.format(\"str %s\", toString());", - " String.format(Locale.ROOT, \"str %s\", toString());", - "", - " // Flagged but not auto-fixed.", - " String.format(\"{} \" + toString(), \"arg\");", - " String.format(Locale.ROOT, \"{} \" + toString(), \"arg\");", - " }", - "", - " void slf4j() {", - " LOG.debug(\"str {}\", toString());", - " LOG.debug((Marker) null, \"str {}\", toString());", - "", - " // Flagged but not auto-fixed.", - " LOG.debug(\"{} \" + toString(), \"arg\");", - " LOG.debug((Marker) null, \"{} \" + toString(), \"arg\");", - " }", - "}") + """ + import static com.google.common.base.Preconditions.checkArgument; + import static org.assertj.core.api.Assertions.assertThat; + + import java.util.Locale; + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + + class A { + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + void assertj() { + assertThat(0).overridingErrorMessage("%s str", toString()); + assertThat(0).overridingErrorMessage("str %s", toString()); + assertThat(0).overridingErrorMessage("%s%s", toString(), toString()); + assertThat(0).overridingErrorMessage("str %s word %s", toString(), new A().hashCode()); + assertThat(0).overridingErrorMessage("str %s word %s", toString(), hashCode() / 2); + + // Flagged but not auto-fixed. + assertThat(0).overridingErrorMessage("%s " + toString(), "arg"); + } + + void guava() { + checkArgument(true, "str %s", toString()); + + // Flagged but not auto-fixed. + checkArgument(true, "%s " + toString(), "arg"); + } + + void jdk() { + String.format("str %s", toString()); + String.format(Locale.ROOT, "str %s", toString()); + + // Flagged but not auto-fixed. + String.format("{} " + toString(), "arg"); + String.format(Locale.ROOT, "{} " + toString(), "arg"); + } + + void slf4j() { + LOG.debug("str {}", toString()); + LOG.debug((Marker) null, "str {}", toString()); + + // Flagged but not auto-fixed. + LOG.debug("{} " + toString(), "arg"); + LOG.debug((Marker) null, "{} " + toString(), "arg"); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/IdentityConversionTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/IdentityConversionTest.java index a8f867502a5..3c88200ff59 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/IdentityConversionTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/IdentityConversionTest.java @@ -17,148 +17,151 @@ void identification() { compilationTestHelper .addSourceLines( "Foo.java", - "import com.google.common.collect.ImmutableBiMap;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableListMultimap;", - "import com.google.common.collect.ImmutableMap;", - "import com.google.common.collect.ImmutableMultimap;", - "import com.google.common.collect.ImmutableMultiset;", - "import com.google.common.collect.ImmutableRangeMap;", - "import com.google.common.collect.ImmutableRangeSet;", - "import com.google.common.collect.ImmutableSet;", - "import com.google.common.collect.ImmutableSetMultimap;", - "import com.google.common.collect.ImmutableTable;", - "import reactor.adapter.rxjava.RxJava2Adapter;", - "import reactor.core.publisher.Flux;", - "import reactor.core.publisher.Mono;", - "", - "public final class Foo {", - " public void foo() {", - " // BUG: Diagnostic contains:", - " Boolean b1 = Boolean.valueOf(Boolean.FALSE);", - " // BUG: Diagnostic contains:", - " Boolean b2 = Boolean.valueOf(false);", - " // BUG: Diagnostic contains:", - " boolean b3 = Boolean.valueOf(Boolean.FALSE);", - " // BUG: Diagnostic contains:", - " boolean b4 = Boolean.valueOf(false);", - "", - " // BUG: Diagnostic contains:", - " Byte byte1 = Byte.valueOf((Byte) Byte.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " Byte byte2 = Byte.valueOf(Byte.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " byte byte3 = Byte.valueOf((Byte) Byte.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " byte byte4 = Byte.valueOf(Byte.MIN_VALUE);", - "", - " // BUG: Diagnostic contains:", - " Character c1 = Character.valueOf((Character) 'a');", - " // BUG: Diagnostic contains:", - " Character c2 = Character.valueOf('a');", - " // BUG: Diagnostic contains:", - " char c3 = Character.valueOf((Character) 'a');", - " // BUG: Diagnostic contains:", - " char c4 = Character.valueOf('a');", - "", - " // BUG: Diagnostic contains:", - " Double d1 = Double.valueOf((Double) 0.0);", - " // BUG: Diagnostic contains:", - " Double d2 = Double.valueOf(0.0);", - " // BUG: Diagnostic contains:", - " double d3 = Double.valueOf((Double) 0.0);", - " // BUG: Diagnostic contains:", - " double d4 = Double.valueOf(0.0);", - "", - " // BUG: Diagnostic contains:", - " Float f1 = Float.valueOf((Float) 0.0F);", - " // BUG: Diagnostic contains:", - " Float f2 = Float.valueOf(0.0F);", - " // BUG: Diagnostic contains:", - " float f3 = Float.valueOf((Float) 0.0F);", - " // BUG: Diagnostic contains:", - " float f4 = Float.valueOf(0.0F);", - "", - " // BUG: Diagnostic contains:", - " Integer i1 = Integer.valueOf((Integer) 1);", - " // BUG: Diagnostic contains:", - " Integer i2 = Integer.valueOf(1);", - " // BUG: Diagnostic contains:", - " int i3 = Integer.valueOf((Integer) 1);", - " // BUG: Diagnostic contains:", - " int i4 = Integer.valueOf(1);", - "", - " // BUG: Diagnostic contains:", - " Long l1 = Long.valueOf((Long) 1L);", - " // BUG: Diagnostic contains:", - " Long l2 = Long.valueOf(1L);", - " // BUG: Diagnostic contains:", - " long l3 = Long.valueOf((Long) 1L);", - " // BUG: Diagnostic contains:", - " long l4 = Long.valueOf(1L);", - "", - " Long l5 = Long.valueOf((Integer) 1);", - " Long l6 = Long.valueOf(1);", - " // BUG: Diagnostic contains:", - " long l7 = Long.valueOf((Integer) 1);", - " // BUG: Diagnostic contains:", - " long l8 = Long.valueOf(1);", - "", - " // BUG: Diagnostic contains:", - " Short s1 = Short.valueOf((Short) Short.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " Short s2 = Short.valueOf(Short.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " short s3 = Short.valueOf((Short) Short.MIN_VALUE);", - " // BUG: Diagnostic contains:", - " short s4 = Short.valueOf(Short.MIN_VALUE);", - "", - " String str1 = String.valueOf(0);", - " // BUG: Diagnostic contains:", - " String str2 = String.valueOf(\"1\");", - "", - " // BUG: Diagnostic contains:", - " ImmutableBiMap o1 = ImmutableBiMap.copyOf(ImmutableBiMap.of());", - " // BUG: Diagnostic contains:", - " ImmutableList o2 = ImmutableList.copyOf(ImmutableList.of());", - " ImmutableListMultimap o3 =", - " // BUG: Diagnostic contains:", - " ImmutableListMultimap.copyOf(ImmutableListMultimap.of());", - " // BUG: Diagnostic contains:", - " ImmutableMap o4 = ImmutableMap.copyOf(ImmutableMap.of());", - " // BUG: Diagnostic contains:", - " ImmutableMultimap o5 = ImmutableMultimap.copyOf(ImmutableMultimap.of());", - " // BUG: Diagnostic contains:", - " ImmutableMultiset o6 = ImmutableMultiset.copyOf(ImmutableMultiset.of());", - " // BUG: Diagnostic contains:", - " ImmutableRangeMap o7 = ImmutableRangeMap.copyOf(ImmutableRangeMap.of());", - " // BUG: Diagnostic contains:", - " ImmutableRangeSet o8 = ImmutableRangeSet.copyOf(ImmutableRangeSet.of());", - " // BUG: Diagnostic contains:", - " ImmutableSet o9 = ImmutableSet.copyOf(ImmutableSet.of());", - " ImmutableSetMultimap o10 =", - " // BUG: Diagnostic contains:", - " ImmutableSetMultimap.copyOf(ImmutableSetMultimap.of());", - " // BUG: Diagnostic contains:", - " ImmutableTable o11 = ImmutableTable.copyOf(ImmutableTable.of());", - "", - " // BUG: Diagnostic contains:", - " Flux flux1 = Flux.just(1).flatMap(e -> RxJava2Adapter.fluxToFlowable(Flux.just(2)));", - " // BUG: Diagnostic contains:", - " Flux flux2 = Flux.concat(Flux.just(1));", - " // BUG: Diagnostic contains:", - " Flux flux3 = Flux.firstWithSignal(Flux.just(1));", - " // BUG: Diagnostic contains:", - " Flux flux4 = Flux.from(Flux.just(1));", - " // BUG: Diagnostic contains:", - " Flux flux5 = Flux.merge(Flux.just(1));", - "", - " // BUG: Diagnostic contains:", - " Mono m1 = Mono.from(Mono.just(1));", - " // BUG: Diagnostic contains:", - " Mono m2 = Mono.fromDirect(Mono.just(1));", - " }", - "}") + """ + import com.google.common.collect.ImmutableBiMap; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableListMultimap; + import com.google.common.collect.ImmutableMap; + import com.google.common.collect.ImmutableMultimap; + import com.google.common.collect.ImmutableMultiset; + import com.google.common.collect.ImmutableRangeMap; + import com.google.common.collect.ImmutableRangeSet; + import com.google.common.collect.ImmutableSet; + import com.google.common.collect.ImmutableSetMultimap; + import com.google.common.collect.ImmutableTable; + import reactor.adapter.rxjava.RxJava2Adapter; + import reactor.core.publisher.Flux; + import reactor.core.publisher.Mono; + + public final class Foo { + public void foo() { + // BUG: Diagnostic contains: + Boolean b1 = Boolean.valueOf(Boolean.FALSE); + // BUG: Diagnostic contains: + Boolean b2 = Boolean.valueOf(false); + // BUG: Diagnostic contains: + boolean b3 = Boolean.valueOf(Boolean.FALSE); + // BUG: Diagnostic contains: + boolean b4 = Boolean.valueOf(false); + + // BUG: Diagnostic contains: + Byte byte1 = Byte.valueOf((Byte) Byte.MIN_VALUE); + // BUG: Diagnostic contains: + Byte byte2 = Byte.valueOf(Byte.MIN_VALUE); + // BUG: Diagnostic contains: + byte byte3 = Byte.valueOf((Byte) Byte.MIN_VALUE); + // BUG: Diagnostic contains: + byte byte4 = Byte.valueOf(Byte.MIN_VALUE); + + // BUG: Diagnostic contains: + Character c1 = Character.valueOf((Character) 'a'); + // BUG: Diagnostic contains: + Character c2 = Character.valueOf('a'); + // BUG: Diagnostic contains: + char c3 = Character.valueOf((Character) 'a'); + // BUG: Diagnostic contains: + char c4 = Character.valueOf('a'); + + // BUG: Diagnostic contains: + Double d1 = Double.valueOf((Double) 0.0); + // BUG: Diagnostic contains: + Double d2 = Double.valueOf(0.0); + // BUG: Diagnostic contains: + double d3 = Double.valueOf((Double) 0.0); + // BUG: Diagnostic contains: + double d4 = Double.valueOf(0.0); + + // BUG: Diagnostic contains: + Float f1 = Float.valueOf((Float) 0.0F); + // BUG: Diagnostic contains: + Float f2 = Float.valueOf(0.0F); + // BUG: Diagnostic contains: + float f3 = Float.valueOf((Float) 0.0F); + // BUG: Diagnostic contains: + float f4 = Float.valueOf(0.0F); + + // BUG: Diagnostic contains: + Integer i1 = Integer.valueOf((Integer) 1); + // BUG: Diagnostic contains: + Integer i2 = Integer.valueOf(1); + // BUG: Diagnostic contains: + int i3 = Integer.valueOf((Integer) 1); + // BUG: Diagnostic contains: + int i4 = Integer.valueOf(1); + + // BUG: Diagnostic contains: + Long l1 = Long.valueOf((Long) 1L); + // BUG: Diagnostic contains: + Long l2 = Long.valueOf(1L); + // BUG: Diagnostic contains: + long l3 = Long.valueOf((Long) 1L); + // BUG: Diagnostic contains: + long l4 = Long.valueOf(1L); + + Long l5 = Long.valueOf((Integer) 1); + Long l6 = Long.valueOf(1); + // BUG: Diagnostic contains: + long l7 = Long.valueOf((Integer) 1); + // BUG: Diagnostic contains: + long l8 = Long.valueOf(1); + + // BUG: Diagnostic contains: + Short s1 = Short.valueOf((Short) Short.MIN_VALUE); + // BUG: Diagnostic contains: + Short s2 = Short.valueOf(Short.MIN_VALUE); + // BUG: Diagnostic contains: + short s3 = Short.valueOf((Short) Short.MIN_VALUE); + // BUG: Diagnostic contains: + short s4 = Short.valueOf(Short.MIN_VALUE); + + String str1 = String.valueOf(0); + // BUG: Diagnostic contains: + String str2 = String.valueOf("1"); + + // BUG: Diagnostic contains: + ImmutableBiMap o1 = ImmutableBiMap.copyOf(ImmutableBiMap.of()); + // BUG: Diagnostic contains: + ImmutableList o2 = ImmutableList.copyOf(ImmutableList.of()); + ImmutableListMultimap o3 = + // BUG: Diagnostic contains: + ImmutableListMultimap.copyOf(ImmutableListMultimap.of()); + // BUG: Diagnostic contains: + ImmutableMap o4 = ImmutableMap.copyOf(ImmutableMap.of()); + // BUG: Diagnostic contains: + ImmutableMultimap o5 = ImmutableMultimap.copyOf(ImmutableMultimap.of()); + // BUG: Diagnostic contains: + ImmutableMultiset o6 = ImmutableMultiset.copyOf(ImmutableMultiset.of()); + // BUG: Diagnostic contains: + ImmutableRangeMap o7 = ImmutableRangeMap.copyOf(ImmutableRangeMap.of()); + // BUG: Diagnostic contains: + ImmutableRangeSet o8 = ImmutableRangeSet.copyOf(ImmutableRangeSet.of()); + // BUG: Diagnostic contains: + ImmutableSet o9 = ImmutableSet.copyOf(ImmutableSet.of()); + ImmutableSetMultimap o10 = + // BUG: Diagnostic contains: + ImmutableSetMultimap.copyOf(ImmutableSetMultimap.of()); + // BUG: Diagnostic contains: + ImmutableTable o11 = ImmutableTable.copyOf(ImmutableTable.of()); + + // BUG: Diagnostic contains: + Flux flux1 = Flux.just(1).flatMap(e -> RxJava2Adapter.fluxToFlowable(Flux.just(2))); + // BUG: Diagnostic contains: + Flux flux2 = Flux.concat(Flux.just(1)); + // BUG: Diagnostic contains: + Flux flux3 = Flux.firstWithSignal(Flux.just(1)); + // BUG: Diagnostic contains: + Flux flux4 = Flux.from(Flux.just(1)); + // BUG: Diagnostic contains: + Flux flux5 = Flux.merge(Flux.just(1)); + + // BUG: Diagnostic contains: + Mono m1 = Mono.from(Mono.just(1)); + // BUG: Diagnostic contains: + Mono m2 = Mono.fromDirect(Mono.just(1)); + } + } + + """) .doTest(); } @@ -168,94 +171,100 @@ void replacementFirstSuggestedFix() { .setFixChooser(FixChoosers.FIRST) .addInputLines( "Foo.java", - "import static org.mockito.Mockito.when;", - "", - "import com.google.common.collect.ImmutableCollection;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableSet;", - "import java.util.ArrayList;", - "import java.util.Collection;", - "import org.reactivestreams.Publisher;", - "import reactor.adapter.rxjava.RxJava2Adapter;", - "import reactor.core.publisher.Flux;", - "import reactor.core.publisher.Mono;", - "", - "public final class Foo {", - " public void foo() {", - " ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of());", - " ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of());", - "", - " ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1));", - " ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - "", - " Collection c1 = ImmutableSet.copyOf(ImmutableSet.of(1));", - " Collection c2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - "", - " Flux f1 = Flux.just(1).flatMap(e -> RxJava2Adapter.fluxToFlowable(Flux.just(2)));", - " Flux f2 = Flux.concat(Flux.just(3));", - " Publisher f3 = Flux.firstWithSignal(Flux.just(4));", - " Publisher f4 = Flux.from(Flux.just(5));", - " Publisher f5 = Flux.merge(Flux.just(6));", - "", - " Mono m1 = Mono.from(Mono.just(7));", - " Publisher m2 = Mono.fromDirect(Mono.just(8));", - "", - " bar(Flux.concat(Flux.just(9)));", - " bar(Mono.from(Mono.just(10)));", - "", - " Object o1 = ImmutableSet.copyOf(ImmutableList.of());", - " Object o2 = ImmutableSet.copyOf(ImmutableSet.of());", - "", - " when(\"foo\".contains(\"f\")).thenAnswer(inv -> ImmutableSet.copyOf(ImmutableList.of(1)));", - " }", - "", - " void bar(Publisher publisher) {}", - "}") + """ + import static org.mockito.Mockito.when; + + import com.google.common.collect.ImmutableCollection; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableSet; + import java.util.ArrayList; + import java.util.Collection; + import org.reactivestreams.Publisher; + import reactor.adapter.rxjava.RxJava2Adapter; + import reactor.core.publisher.Flux; + import reactor.core.publisher.Mono; + + public final class Foo { + public void foo() { + ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of()); + ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of()); + + ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1)); + ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + + Collection c1 = ImmutableSet.copyOf(ImmutableSet.of(1)); + Collection c2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + + Flux f1 = Flux.just(1).flatMap(e -> RxJava2Adapter.fluxToFlowable(Flux.just(2))); + Flux f2 = Flux.concat(Flux.just(3)); + Publisher f3 = Flux.firstWithSignal(Flux.just(4)); + Publisher f4 = Flux.from(Flux.just(5)); + Publisher f5 = Flux.merge(Flux.just(6)); + + Mono m1 = Mono.from(Mono.just(7)); + Publisher m2 = Mono.fromDirect(Mono.just(8)); + + bar(Flux.concat(Flux.just(9))); + bar(Mono.from(Mono.just(10))); + + Object o1 = ImmutableSet.copyOf(ImmutableList.of()); + Object o2 = ImmutableSet.copyOf(ImmutableSet.of()); + + when("foo".contains("f")).thenAnswer(inv -> ImmutableSet.copyOf(ImmutableList.of(1))); + } + + void bar(Publisher publisher) {} + } + + """) .addOutputLines( "Foo.java", - "import static org.mockito.Mockito.when;", - "", - "import com.google.common.collect.ImmutableCollection;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableSet;", - "import java.util.ArrayList;", - "import java.util.Collection;", - "import org.reactivestreams.Publisher;", - "import reactor.adapter.rxjava.RxJava2Adapter;", - "import reactor.core.publisher.Flux;", - "import reactor.core.publisher.Mono;", - "", - "public final class Foo {", - " public void foo() {", - " ImmutableSet set1 = ImmutableSet.of();", - " ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of());", - "", - " ImmutableCollection list1 = ImmutableList.of(1);", - " ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - "", - " Collection c1 = ImmutableSet.of(1);", - " Collection c2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - "", - " Flux f1 = Flux.just(1).flatMap(e -> Flux.just(2));", - " Flux f2 = Flux.just(3);", - " Publisher f3 = Flux.just(4);", - " Publisher f4 = Flux.just(5);", - " Publisher f5 = Flux.just(6);", - "", - " Mono m1 = Mono.just(7);", - " Publisher m2 = Mono.just(8);", - "", - " bar(Flux.just(9));", - " bar(Mono.just(10));", - "", - " Object o1 = ImmutableSet.copyOf(ImmutableList.of());", - " Object o2 = ImmutableSet.of();", - "", - " when(\"foo\".contains(\"f\")).thenAnswer(inv -> ImmutableSet.copyOf(ImmutableList.of(1)));", - " }", - "", - " void bar(Publisher publisher) {}", - "}") + """ + import static org.mockito.Mockito.when; + + import com.google.common.collect.ImmutableCollection; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableSet; + import java.util.ArrayList; + import java.util.Collection; + import org.reactivestreams.Publisher; + import reactor.adapter.rxjava.RxJava2Adapter; + import reactor.core.publisher.Flux; + import reactor.core.publisher.Mono; + + public final class Foo { + public void foo() { + ImmutableSet set1 = ImmutableSet.of(); + ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of()); + + ImmutableCollection list1 = ImmutableList.of(1); + ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + + Collection c1 = ImmutableSet.of(1); + Collection c2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + + Flux f1 = Flux.just(1).flatMap(e -> Flux.just(2)); + Flux f2 = Flux.just(3); + Publisher f3 = Flux.just(4); + Publisher f4 = Flux.just(5); + Publisher f5 = Flux.just(6); + + Mono m1 = Mono.just(7); + Publisher m2 = Mono.just(8); + + bar(Flux.just(9)); + bar(Mono.just(10)); + + Object o1 = ImmutableSet.copyOf(ImmutableList.of()); + Object o2 = ImmutableSet.of(); + + when("foo".contains("f")).thenAnswer(inv -> ImmutableSet.copyOf(ImmutableList.of(1))); + } + + void bar(Publisher publisher) {} + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -265,38 +274,44 @@ void replacementSecondSuggestedFix() { .setFixChooser(FixChoosers.SECOND) .addInputLines( "Foo.java", - "import com.google.common.collect.ImmutableCollection;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableSet;", - "import java.util.ArrayList;", - "", - "public final class Foo {", - " public void foo() {", - " ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of());", - " ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of());", - "", - " ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1));", - " ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - " }", - "}") + """ + import com.google.common.collect.ImmutableCollection; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableSet; + import java.util.ArrayList; + + public final class Foo { + public void foo() { + ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of()); + ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of()); + + ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1)); + ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + } + } + + """) .addOutputLines( "Foo.java", - "import com.google.common.collect.ImmutableCollection;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableSet;", - "import java.util.ArrayList;", - "", - "public final class Foo {", - " public void foo() {", - " @SuppressWarnings(\"IdentityConversion\")", - " ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of());", - " ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of());", - "", - " @SuppressWarnings(\"IdentityConversion\")", - " ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1));", - " ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1)));", - " }", - "}") + """ + import com.google.common.collect.ImmutableCollection; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableSet; + import java.util.ArrayList; + + public final class Foo { + public void foo() { + @SuppressWarnings("IdentityConversion") + ImmutableSet set1 = ImmutableSet.copyOf(ImmutableSet.of()); + ImmutableSet set2 = ImmutableSet.copyOf(ImmutableList.of()); + + @SuppressWarnings("IdentityConversion") + ImmutableCollection list1 = ImmutableList.copyOf(ImmutableList.of(1)); + ImmutableCollection list2 = ImmutableList.copyOf(new ArrayList<>(ImmutableList.of(1))); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/JUnitMethodDeclarationTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/JUnitMethodDeclarationTest.java index e0f7e634923..63c625b820f 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/JUnitMethodDeclarationTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/JUnitMethodDeclarationTest.java @@ -16,291 +16,300 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static org.junit.jupiter.params.provider.Arguments.arguments;", - "", - "import org.junit.jupiter.api.AfterAll;", - "import org.junit.jupiter.api.AfterEach;", - "import org.junit.jupiter.api.BeforeAll;", - "import org.junit.jupiter.api.BeforeEach;", - "import org.junit.jupiter.api.Test;", - "import org.junit.jupiter.params.ParameterizedTest;", - "", - "class A {", - " {", - " arguments();", - " }", - "", - " @BeforeAll", - " void setUp1() {}", - "", - " @BeforeAll", - " // BUG: Diagnostic contains:", - " public void setUp2() {}", - "", - " @BeforeAll", - " // BUG: Diagnostic contains:", - " protected void setUp3() {}", - "", - " @BeforeAll", - " // BUG: Diagnostic contains:", - " private void setUp4() {}", - "", - " @BeforeEach", - " void setup5() {}", - "", - " @BeforeEach", - " // BUG: Diagnostic contains:", - " public void setUp6() {}", - "", - " @BeforeEach", - " // BUG: Diagnostic contains:", - " protected void setUp7() {}", - "", - " @BeforeEach", - " // BUG: Diagnostic contains:", - " private void setUp8() {}", - "", - " @AfterEach", - " void tearDown1() {}", - "", - " @AfterEach", - " // BUG: Diagnostic contains:", - " public void tearDown2() {}", - "", - " @AfterEach", - " // BUG: Diagnostic contains:", - " protected void tearDown3() {}", - "", - " @AfterEach", - " // BUG: Diagnostic contains:", - " private void tearDown4() {}", - "", - " @AfterAll", - " void tearDown5() {}", - "", - " @AfterAll", - " // BUG: Diagnostic contains:", - " public void tearDown6() {}", - "", - " @AfterAll", - " // BUG: Diagnostic contains:", - " protected void tearDown7() {}", - "", - " @AfterAll", - " // BUG: Diagnostic contains:", - " private void tearDown8() {}", - "", - " @Test", - " void method1() {}", - "", - " @Test", - " // BUG: Diagnostic contains:", - " void testMethod2() {}", - "", - " @Test", - " // BUG: Diagnostic contains:", - " public void method3() {}", - "", - " @Test", - " // BUG: Diagnostic contains:", - " protected void method4() {}", - "", - " @Test", - " // BUG: Diagnostic contains:", - " private void method5() {}", - "", - " @ParameterizedTest", - " void method6() {}", - "", - " @ParameterizedTest", - " // BUG: Diagnostic contains:", - " void testMethod7() {}", - "", - " @ParameterizedTest", - " // BUG: Diagnostic contains:", - " public void method8() {}", - "", - " @ParameterizedTest", - " // BUG: Diagnostic contains:", - " protected void method9() {}", - "", - " @ParameterizedTest", - " // BUG: Diagnostic contains:", - " private void method10() {}", - "", - " @BeforeEach", - " @BeforeAll", - " @AfterEach", - " @AfterAll", - " void testNonTestMethod1() {}", - "", - " public void testNonTestMethod2() {}", - "", - " protected void testNonTestMethod3() {}", - "", - " private void testNonTestMethod4() {}", - "", - " @Test", - " void test5() {}", - "", - " @Test", - " // BUG: Diagnostic contains: (but note that a method named `overload` already exists in this", - " // class)", - " void testOverload() {}", - "", - " void overload() {}", - "", - " @Test", - " // BUG: Diagnostic contains: (but note that `arguments` is already statically imported)", - " void testArguments() {}", - "", - " @Test", - " // BUG: Diagnostic contains: (but note that `public` is a reserved keyword)", - " void testPublic() {}", - "}") + """ + import static org.junit.jupiter.params.provider.Arguments.arguments; + + import org.junit.jupiter.api.AfterAll; + import org.junit.jupiter.api.AfterEach; + import org.junit.jupiter.api.BeforeAll; + import org.junit.jupiter.api.BeforeEach; + import org.junit.jupiter.api.Test; + import org.junit.jupiter.params.ParameterizedTest; + + class A { + { + arguments(); + } + + @BeforeAll + void setUp1() {} + + @BeforeAll + // BUG: Diagnostic contains: + public void setUp2() {} + + @BeforeAll + // BUG: Diagnostic contains: + protected void setUp3() {} + + @BeforeAll + // BUG: Diagnostic contains: + private void setUp4() {} + + @BeforeEach + void setup5() {} + + @BeforeEach + // BUG: Diagnostic contains: + public void setUp6() {} + + @BeforeEach + // BUG: Diagnostic contains: + protected void setUp7() {} + + @BeforeEach + // BUG: Diagnostic contains: + private void setUp8() {} + + @AfterEach + void tearDown1() {} + + @AfterEach + // BUG: Diagnostic contains: + public void tearDown2() {} + + @AfterEach + // BUG: Diagnostic contains: + protected void tearDown3() {} + + @AfterEach + // BUG: Diagnostic contains: + private void tearDown4() {} + + @AfterAll + void tearDown5() {} + + @AfterAll + // BUG: Diagnostic contains: + public void tearDown6() {} + + @AfterAll + // BUG: Diagnostic contains: + protected void tearDown7() {} + + @AfterAll + // BUG: Diagnostic contains: + private void tearDown8() {} + + @Test + void method1() {} + + @Test + // BUG: Diagnostic contains: + void testMethod2() {} + + @Test + // BUG: Diagnostic contains: + public void method3() {} + + @Test + // BUG: Diagnostic contains: + protected void method4() {} + + @Test + // BUG: Diagnostic contains: + private void method5() {} + + @ParameterizedTest + void method6() {} + + @ParameterizedTest + // BUG: Diagnostic contains: + void testMethod7() {} + + @ParameterizedTest + // BUG: Diagnostic contains: + public void method8() {} + + @ParameterizedTest + // BUG: Diagnostic contains: + protected void method9() {} + + @ParameterizedTest + // BUG: Diagnostic contains: + private void method10() {} + + @BeforeEach + @BeforeAll + @AfterEach + @AfterAll + void testNonTestMethod1() {} + + public void testNonTestMethod2() {} + + protected void testNonTestMethod3() {} + + private void testNonTestMethod4() {} + + @Test + void test5() {} + + @Test + // BUG: Diagnostic contains: (but note that a method named `overload` already exists in this + // class) + void testOverload() {} + + void overload() {} + + @Test + // BUG: Diagnostic contains: (but note that `arguments` is already statically imported) + void testArguments() {} + + @Test + // BUG: Diagnostic contains: (but note that `public` is a reserved keyword) + void testPublic() {} + } + + """) .addSourceLines( "B.java", - "import org.junit.jupiter.api.AfterAll;", - "import org.junit.jupiter.api.AfterEach;", - "import org.junit.jupiter.api.BeforeAll;", - "import org.junit.jupiter.api.BeforeEach;", - "import org.junit.jupiter.api.Test;", - "import org.junit.jupiter.params.ParameterizedTest;", - "", - "class B extends A {", - " @Override", - " @BeforeAll", - " void setUp1() {}", - "", - " @Override", - " @BeforeAll", - " public void setUp2() {}", - "", - " @Override", - " @BeforeAll", - " protected void setUp3() {}", - "", - " @Override", - " @BeforeEach", - " void setup5() {}", - "", - " @Override", - " @BeforeEach", - " public void setUp6() {}", - "", - " @Override", - " @BeforeEach", - " protected void setUp7() {}", - "", - " @Override", - " @AfterEach", - " void tearDown1() {}", - "", - " @Override", - " @AfterEach", - " public void tearDown2() {}", - "", - " @Override", - " @AfterEach", - " protected void tearDown3() {}", - "", - " @Override", - " @AfterAll", - " void tearDown5() {}", - "", - " @Override", - " @AfterAll", - " public void tearDown6() {}", - "", - " @Override", - " @AfterAll", - " protected void tearDown7() {}", - "", - " @Override", - " @Test", - " void method1() {}", - "", - " @Override", - " @Test", - " void testMethod2() {}", - "", - " @Override", - " @Test", - " public void method3() {}", - "", - " @Override", - " @Test", - " protected void method4() {}", - "", - " @Override", - " @ParameterizedTest", - " void method6() {}", - "", - " @Override", - " @ParameterizedTest", - " void testMethod7() {}", - "", - " @Override", - " @ParameterizedTest", - " public void method8() {}", - "", - " @Override", - " @ParameterizedTest", - " protected void method9() {}", - "", - " @Override", - " @BeforeEach", - " @BeforeAll", - " @AfterEach", - " @AfterAll", - " void testNonTestMethod1() {}", - "", - " @Override", - " public void testNonTestMethod2() {}", - "", - " @Override", - " protected void testNonTestMethod3() {}", - "", - " @Override", - " @Test", - " void test5() {}", - "", - " @Override", - " @Test", - " void testOverload() {}", - "", - " @Override", - " void overload() {}", - "", - " @Override", - " @Test", - " void testArguments() {}", - "", - " @Override", - " @Test", - " void testPublic() {}", - "}") + """ + import org.junit.jupiter.api.AfterAll; + import org.junit.jupiter.api.AfterEach; + import org.junit.jupiter.api.BeforeAll; + import org.junit.jupiter.api.BeforeEach; + import org.junit.jupiter.api.Test; + import org.junit.jupiter.params.ParameterizedTest; + + class B extends A { + @Override + @BeforeAll + void setUp1() {} + + @Override + @BeforeAll + public void setUp2() {} + + @Override + @BeforeAll + protected void setUp3() {} + + @Override + @BeforeEach + void setup5() {} + + @Override + @BeforeEach + public void setUp6() {} + + @Override + @BeforeEach + protected void setUp7() {} + + @Override + @AfterEach + void tearDown1() {} + + @Override + @AfterEach + public void tearDown2() {} + + @Override + @AfterEach + protected void tearDown3() {} + + @Override + @AfterAll + void tearDown5() {} + + @Override + @AfterAll + public void tearDown6() {} + + @Override + @AfterAll + protected void tearDown7() {} + + @Override + @Test + void method1() {} + + @Override + @Test + void testMethod2() {} + + @Override + @Test + public void method3() {} + + @Override + @Test + protected void method4() {} + + @Override + @ParameterizedTest + void method6() {} + + @Override + @ParameterizedTest + void testMethod7() {} + + @Override + @ParameterizedTest + public void method8() {} + + @Override + @ParameterizedTest + protected void method9() {} + + @Override + @BeforeEach + @BeforeAll + @AfterEach + @AfterAll + void testNonTestMethod1() {} + + @Override + public void testNonTestMethod2() {} + + @Override + protected void testNonTestMethod3() {} + + @Override + @Test + void test5() {} + + @Override + @Test + void testOverload() {} + + @Override + void overload() {} + + @Override + @Test + void testArguments() {} + + @Override + @Test + void testPublic() {} + } + + """) .addSourceLines( "C.java", - "import org.junit.jupiter.api.AfterAll;", - "import org.junit.jupiter.api.BeforeAll;", - "import org.junit.jupiter.api.Test;", - "", - "abstract class C {", - " @BeforeAll", - " public void setUp() {}", - "", - " @Test", - " void testMethod1() {}", - "", - " @AfterAll", - " // BUG: Diagnostic contains:", - " private void tearDown() {}", - "", - " @Test", - " // BUG: Diagnostic contains:", - " final void testMethod2() {}", - "}") + """ + import org.junit.jupiter.api.AfterAll; + import org.junit.jupiter.api.BeforeAll; + import org.junit.jupiter.api.Test; + + abstract class C { + @BeforeAll + public void setUp() {} + + @Test + void testMethod1() {} + + @AfterAll + // BUG: Diagnostic contains: + private void tearDown() {} + + @Test + // BUG: Diagnostic contains: + final void testMethod2() {} + } + + """) .doTest(); } @@ -309,114 +318,120 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static org.junit.jupiter.params.provider.Arguments.arguments;", - "", - "import org.junit.jupiter.api.AfterAll;", - "import org.junit.jupiter.api.AfterEach;", - "import org.junit.jupiter.api.BeforeAll;", - "import org.junit.jupiter.api.BeforeEach;", - "import org.junit.jupiter.api.RepeatedTest;", - "import org.junit.jupiter.api.Test;", - "import org.junit.jupiter.params.ParameterizedTest;", - "", - "class A {", - " {", - " arguments();", - " }", - "", - " @BeforeAll", - " public void setUp1() {}", - "", - " @BeforeEach", - " protected void setUp2() {}", - "", - " @AfterEach", - " private void setUp3() {}", - "", - " @AfterAll", - " private void setUp4() {}", - "", - " @Test", - " void testFoo() {}", - "", - " @ParameterizedTest", - " void testBar() {}", - "", - " @Test", - " public void baz() {}", - "", - " @RepeatedTest(2)", - " private void qux() {}", - "", - " @ParameterizedTest", - " protected void quux() {}", - "", - " @Test", - " public void testOverload() {}", - "", - " void overload() {}", - "", - " @Test", - " protected void testArguments() {}", - "", - " @Test", - " private void testClass() {}", - "}") + """ + import static org.junit.jupiter.params.provider.Arguments.arguments; + + import org.junit.jupiter.api.AfterAll; + import org.junit.jupiter.api.AfterEach; + import org.junit.jupiter.api.BeforeAll; + import org.junit.jupiter.api.BeforeEach; + import org.junit.jupiter.api.RepeatedTest; + import org.junit.jupiter.api.Test; + import org.junit.jupiter.params.ParameterizedTest; + + class A { + { + arguments(); + } + + @BeforeAll + public void setUp1() {} + + @BeforeEach + protected void setUp2() {} + + @AfterEach + private void setUp3() {} + + @AfterAll + private void setUp4() {} + + @Test + void testFoo() {} + + @ParameterizedTest + void testBar() {} + + @Test + public void baz() {} + + @RepeatedTest(2) + private void qux() {} + + @ParameterizedTest + protected void quux() {} + + @Test + public void testOverload() {} + + void overload() {} + + @Test + protected void testArguments() {} + + @Test + private void testClass() {} + } + + """) .addOutputLines( "out/A.java", - "import static org.junit.jupiter.params.provider.Arguments.arguments;", - "", - "import org.junit.jupiter.api.AfterAll;", - "import org.junit.jupiter.api.AfterEach;", - "import org.junit.jupiter.api.BeforeAll;", - "import org.junit.jupiter.api.BeforeEach;", - "import org.junit.jupiter.api.RepeatedTest;", - "import org.junit.jupiter.api.Test;", - "import org.junit.jupiter.params.ParameterizedTest;", - "", - "class A {", - " {", - " arguments();", - " }", - "", - " @BeforeAll", - " void setUp1() {}", - "", - " @BeforeEach", - " void setUp2() {}", - "", - " @AfterEach", - " void setUp3() {}", - "", - " @AfterAll", - " void setUp4() {}", - "", - " @Test", - " void foo() {}", - "", - " @ParameterizedTest", - " void bar() {}", - "", - " @Test", - " void baz() {}", - "", - " @RepeatedTest(2)", - " void qux() {}", - "", - " @ParameterizedTest", - " void quux() {}", - "", - " @Test", - " void testOverload() {}", - "", - " void overload() {}", - "", - " @Test", - " void testArguments() {}", - "", - " @Test", - " void testClass() {}", - "}") + """ + import static org.junit.jupiter.params.provider.Arguments.arguments; + + import org.junit.jupiter.api.AfterAll; + import org.junit.jupiter.api.AfterEach; + import org.junit.jupiter.api.BeforeAll; + import org.junit.jupiter.api.BeforeEach; + import org.junit.jupiter.api.RepeatedTest; + import org.junit.jupiter.api.Test; + import org.junit.jupiter.params.ParameterizedTest; + + class A { + { + arguments(); + } + + @BeforeAll + void setUp1() {} + + @BeforeEach + void setUp2() {} + + @AfterEach + void setUp3() {} + + @AfterAll + void setUp4() {} + + @Test + void foo() {} + + @ParameterizedTest + void bar() {} + + @Test + void baz() {} + + @RepeatedTest(2) + void qux() {} + + @ParameterizedTest + void quux() {} + + @Test + void testOverload() {} + + void overload() {} + + @Test + void testArguments() {} + + @Test + void testClass() {} + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationAttributeListingTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationAttributeListingTest.java index 24191fa1f4e..8e4dd19a4c6 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationAttributeListingTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationAttributeListingTest.java @@ -25,127 +25,130 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static java.math.RoundingMode.DOWN;", - "import static java.math.RoundingMode.UP;", - "", - "import com.fasterxml.jackson.annotation.JsonPropertyOrder;", - "import io.swagger.annotations.ApiImplicitParam;", - "import io.swagger.annotations.ApiImplicitParams;", - "import io.swagger.v3.oas.annotations.Parameter;", - "import io.swagger.v3.oas.annotations.Parameters;", - "import java.math.RoundingMode;", - "import javax.xml.bind.annotation.XmlType;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] ints() default {};", - "", - " Class[] cls() default {};", - "", - " RoundingMode[] enums() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @Foo({})", - " A noString();", - "", - " @Foo({\"a\"})", - " A oneString();", - "", - " @Foo({\"a\", \"b\"})", - " A sortedStrings();", - " // BUG: Diagnostic contains:", - " @Foo({\"b\", \"a\"})", - " A unsortedString();", - "", - " @Foo({\"ab\", \"Ac\"})", - " A sortedStringCaseInsensitive();", - " // BUG: Diagnostic contains:", - " @Foo({\"ac\", \"Ab\"})", - " A unsortedStringCaseInsensitive();", - "", - " @Foo({\"A\", \"a\"})", - " A sortedStringCaseInsensitiveWithTotalOrderFallback();", - " // BUG: Diagnostic contains:", - " @Foo({\"a\", \"A\"})", - " A unsortedStringCaseInsensitiveWithTotalOrderFallback();", - "", - " @Foo(ints = {})", - " A noInts();", - "", - " @Foo(ints = {0})", - " A oneInt();", - "", - " @Foo(ints = {0, 1})", - " A sortedInts();", - "", - " @Foo(ints = {1, 0})", - " A unsortedInts();", - "", - " @Foo(cls = {})", - " A noClasses();", - "", - " @Foo(cls = {int.class})", - " A oneClass();", - "", - " @Foo(cls = {int.class, long.class})", - " A sortedClasses();", - " // BUG: Diagnostic contains:", - " @Foo(cls = {long.class, int.class})", - " A unsortedClasses();", - "", - " @Foo(enums = {})", - " A noEnums();", - "", - " @Foo(enums = {DOWN})", - " A oneEnum();", - "", - " @Foo(enums = {DOWN, UP})", - " A sortedEnums();", - " // BUG: Diagnostic contains:", - " @Foo(enums = {UP, DOWN})", - " A unsortedEnums();", - "", - " @Foo(anns = {})", - " A noAnns();", - "", - " @Foo(anns = {@Bar(\"a\")})", - " A oneAnn();", - "", - " @Foo(anns = {@Bar(\"a\"), @Bar(\"b\")})", - " A sortedAnns();", - " // BUG: Diagnostic contains:", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " A unsortedAnns();", - " // BUG: Diagnostic contains:", - " @Foo(anns = {@Bar(\"a\"), @Bar({\"b\", \"a\"})})", - " A unsortedInnderAnns();", - "", - " @Foo({\"a=foo\", \"a.b=bar\", \"a.c=baz\"})", - " A hierarchicallySorted();", - " // BUG: Diagnostic contains:", - " @Foo({\"a.b=bar\", \"a.c=baz\", \"a=foo\"})", - " A hierarchicallyUnsorted();", - "", - " @JsonPropertyOrder({\"field2\", \"field1\"})", - " A dto();", - "", - " @ApiImplicitParams({@ApiImplicitParam(\"p2\"), @ApiImplicitParam(\"p1\")})", - " A firstEndpoint();", - "", - " @Parameters({@Parameter(name = \"p2\"), @Parameter(name = \"p1\")})", - " A secondEndpoint();", - "", - " @XmlType(propOrder = {\"field2\", \"field1\"})", - " class Dummy {}", - "}") + """ + import static java.math.RoundingMode.DOWN; + import static java.math.RoundingMode.UP; + + import com.fasterxml.jackson.annotation.JsonPropertyOrder; + import io.swagger.annotations.ApiImplicitParam; + import io.swagger.annotations.ApiImplicitParams; + import io.swagger.v3.oas.annotations.Parameter; + import io.swagger.v3.oas.annotations.Parameters; + import java.math.RoundingMode; + import javax.xml.bind.annotation.XmlType; + + interface A { + @interface Foo { + String[] value() default {}; + + int[] ints() default {}; + + Class[] cls() default {}; + + RoundingMode[] enums() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @Foo({}) + A noString(); + + @Foo({"a"}) + A oneString(); + + @Foo({"a", "b"}) + A sortedStrings(); + // BUG: Diagnostic contains: + @Foo({"b", "a"}) + A unsortedString(); + + @Foo({"ab", "Ac"}) + A sortedStringCaseInsensitive(); + // BUG: Diagnostic contains: + @Foo({"ac", "Ab"}) + A unsortedStringCaseInsensitive(); + + @Foo({"A", "a"}) + A sortedStringCaseInsensitiveWithTotalOrderFallback(); + // BUG: Diagnostic contains: + @Foo({"a", "A"}) + A unsortedStringCaseInsensitiveWithTotalOrderFallback(); + + @Foo(ints = {}) + A noInts(); + + @Foo(ints = {0}) + A oneInt(); + + @Foo(ints = {0, 1}) + A sortedInts(); + + @Foo(ints = {1, 0}) + A unsortedInts(); + + @Foo(cls = {}) + A noClasses(); + + @Foo(cls = {int.class}) + A oneClass(); + + @Foo(cls = {int.class, long.class}) + A sortedClasses(); + // BUG: Diagnostic contains: + @Foo(cls = {long.class, int.class}) + A unsortedClasses(); + + @Foo(enums = {}) + A noEnums(); + + @Foo(enums = {DOWN}) + A oneEnum(); + + @Foo(enums = {DOWN, UP}) + A sortedEnums(); + // BUG: Diagnostic contains: + @Foo(enums = {UP, DOWN}) + A unsortedEnums(); + + @Foo(anns = {}) + A noAnns(); + + @Foo(anns = {@Bar("a")}) + A oneAnn(); + + @Foo(anns = {@Bar("a"), @Bar("b")}) + A sortedAnns(); + // BUG: Diagnostic contains: + @Foo(anns = {@Bar("b"), @Bar("a")}) + A unsortedAnns(); + // BUG: Diagnostic contains: + @Foo(anns = {@Bar("a"), @Bar({"b", "a"})}) + A unsortedInnderAnns(); + + @Foo({"a=foo", "a.b=bar", "a.c=baz"}) + A hierarchicallySorted(); + // BUG: Diagnostic contains: + @Foo({"a.b=bar", "a.c=baz", "a=foo"}) + A hierarchicallyUnsorted(); + + @JsonPropertyOrder({"field2", "field1"}) + A dto(); + + @ApiImplicitParams({@ApiImplicitParam("p2"), @ApiImplicitParam("p1")}) + A firstEndpoint(); + + @Parameters({@Parameter(name = "p2"), @Parameter(name = "p1")}) + A secondEndpoint(); + + @XmlType(propOrder = {"field2", "field1"}) + class Dummy {} + } + + """) .doTest(); } @@ -157,78 +160,84 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static java.math.RoundingMode.DOWN;", - "import static java.math.RoundingMode.UP;", - "", - "import java.math.RoundingMode;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " Class[] cls() default {};", - "", - " RoundingMode[] enums() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @Foo({\"b\", \"a\"})", - " A unsortedString();", - "", - " @Foo(cls = {long.class, int.class})", - " A unsortedClasses();", - "", - " @Foo(enums = {UP, DOWN})", - " A unsortedEnums();", - "", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " A unsortedAnns();", - "", - " @Foo(anns = {@Bar(\"a\"), @Bar({\"b\", \"a\"})})", - " A unsortedInnderAnns();", - "}") + """ + import static java.math.RoundingMode.DOWN; + import static java.math.RoundingMode.UP; + + import java.math.RoundingMode; + + interface A { + @interface Foo { + String[] value() default {}; + + Class[] cls() default {}; + + RoundingMode[] enums() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @Foo({"b", "a"}) + A unsortedString(); + + @Foo(cls = {long.class, int.class}) + A unsortedClasses(); + + @Foo(enums = {UP, DOWN}) + A unsortedEnums(); + + @Foo(anns = {@Bar("b"), @Bar("a")}) + A unsortedAnns(); + + @Foo(anns = {@Bar("a"), @Bar({"b", "a"})}) + A unsortedInnderAnns(); + } + + """) .addOutputLines( "out/A.java", - "import static java.math.RoundingMode.DOWN;", - "import static java.math.RoundingMode.UP;", - "", - "import java.math.RoundingMode;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " Class[] cls() default {};", - "", - " RoundingMode[] enums() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @Foo({\"a\", \"b\"})", - " A unsortedString();", - "", - " @Foo(cls = {int.class, long.class})", - " A unsortedClasses();", - "", - " @Foo(enums = {DOWN, UP})", - " A unsortedEnums();", - "", - " @Foo(anns = {@Bar(\"a\"), @Bar(\"b\")})", - " A unsortedAnns();", - "", - " @Foo(anns = {@Bar(\"a\"), @Bar({\"a\", \"b\"})})", - " A unsortedInnderAnns();", - "}") + """ + import static java.math.RoundingMode.DOWN; + import static java.math.RoundingMode.UP; + + import java.math.RoundingMode; + + interface A { + @interface Foo { + String[] value() default {}; + + Class[] cls() default {}; + + RoundingMode[] enums() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @Foo({"a", "b"}) + A unsortedString(); + + @Foo(cls = {int.class, long.class}) + A unsortedClasses(); + + @Foo(enums = {DOWN, UP}) + A unsortedEnums(); + + @Foo(anns = {@Bar("a"), @Bar("b")}) + A unsortedAnns(); + + @Foo(anns = {@Bar("a"), @Bar({"a", "b"})}) + A unsortedInnderAnns(); + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -238,46 +247,49 @@ void filtering() { restrictedCompilationTestHelper .addSourceLines( "pkg/A.java", - "package pkg;", - "", - "interface A {", - " @interface Foo {", - " String[] value() default {};", - "", - " String[] value2() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - "", - " String[] value2() default {};", - " }", - "", - " @interface Baz {", - " String[] value() default {};", - "", - " String[] value2() default {};", - " }", - "", - " // BUG: Diagnostic contains:", - " @Foo({\"b\", \"a\"})", - " A fooValue();", - " // BUG: Diagnostic contains:", - " @Foo(value2 = {\"b\", \"a\"})", - " A fooValue2();", - "", - " @Bar({\"b\", \"a\"})", - " A barValue();", - " // BUG: Diagnostic contains:", - " @Bar(value2 = {\"b\", \"a\"})", - " A barValue2();", - "", - " @Baz({\"b\", \"a\"})", - " A bazValue();", - "", - " @Baz(value2 = {\"b\", \"a\"})", - " A bazValue2();", - "}") + """ + package pkg; + + interface A { + @interface Foo { + String[] value() default {}; + + String[] value2() default {}; + } + + @interface Bar { + String[] value() default {}; + + String[] value2() default {}; + } + + @interface Baz { + String[] value() default {}; + + String[] value2() default {}; + } + + // BUG: Diagnostic contains: + @Foo({"b", "a"}) + A fooValue(); + // BUG: Diagnostic contains: + @Foo(value2 = {"b", "a"}) + A fooValue2(); + + @Bar({"b", "a"}) + A barValue(); + // BUG: Diagnostic contains: + @Bar(value2 = {"b", "a"}) + A barValue2(); + + @Baz({"b", "a"}) + A bazValue(); + + @Baz(value2 = {"b", "a"}) + A bazValue2(); + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationListingTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationListingTest.java index 3af843564e8..aecc1778c48 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationListingTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/LexicographicalAnnotationListingTest.java @@ -21,92 +21,95 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import java.lang.annotation.Repeatable;", - "", - "interface A {", - " @Repeatable(Foos.class)", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] ints() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @interface Baz {", - " String[] str() default {};", - " }", - "", - " @interface Foos {", - " Foo[] value();", - " }", - "", - " // BUG: Diagnostic matches: X", - " @Foo", - " @Bar", - " A unsortedSimpleCase();", - " // BUG: Diagnostic matches: X", - " @Foo()", - " @Bar()", - " A unsortedWithParens();", - "", - " @Foo()", - " A onlyOneAnnotation();", - "", - " @Bar", - " @Foo()", - " A sortedAnnotationsOneWithParens();", - "", - " // BUG: Diagnostic matches: X", - " @Foo", - " @Baz", - " @Bar", - " A threeUnsortedAnnotationsSameInitialLetter();", - " // BUG: Diagnostic matches: X", - " @Bar", - " @Foo()", - " @Baz", - " A firstOrderedWithTwoUnsortedAnnotations();", - "", - " @Bar", - " @Baz", - " @Foo()", - " A threeSortedAnnotations();", - "", - " // BUG: Diagnostic matches: X", - " @Foo({\"b\"})", - " @Bar({\"a\"})", - " A unsortedWithStringAttributes();", - " // BUG: Diagnostic matches: X", - " @Baz(str = {\"a\", \"b\"})", - " @Foo(ints = {1, 0})", - " @Bar", - " A unsortedWithAttributes();", - " // BUG: Diagnostic matches: X", - " @Bar", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Baz", - " A unsortedWithNestedBar();", - "", - " @Bar", - " @Baz", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " A sortedWithNestedBar();", - "", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Foo(ints = {1, 2})", - " @Foo({\"b\"})", - " A sortedRepeatableAnnotation();", - " // BUG: Diagnostic matches: X", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Bar", - " @Foo(ints = {1, 2})", - " A unsortedRepeatableAnnotation();", - "}") + """ + import java.lang.annotation.Repeatable; + + interface A { + @Repeatable(Foos.class) + @interface Foo { + String[] value() default {}; + + int[] ints() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @interface Baz { + String[] str() default {}; + } + + @interface Foos { + Foo[] value(); + } + + // BUG: Diagnostic matches: X + @Foo + @Bar + A unsortedSimpleCase(); + // BUG: Diagnostic matches: X + @Foo() + @Bar() + A unsortedWithParens(); + + @Foo() + A onlyOneAnnotation(); + + @Bar + @Foo() + A sortedAnnotationsOneWithParens(); + + // BUG: Diagnostic matches: X + @Foo + @Baz + @Bar + A threeUnsortedAnnotationsSameInitialLetter(); + // BUG: Diagnostic matches: X + @Bar + @Foo() + @Baz + A firstOrderedWithTwoUnsortedAnnotations(); + + @Bar + @Baz + @Foo() + A threeSortedAnnotations(); + + // BUG: Diagnostic matches: X + @Foo({"b"}) + @Bar({"a"}) + A unsortedWithStringAttributes(); + // BUG: Diagnostic matches: X + @Baz(str = {"a", "b"}) + @Foo(ints = {1, 0}) + @Bar + A unsortedWithAttributes(); + // BUG: Diagnostic matches: X + @Bar + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Baz + A unsortedWithNestedBar(); + + @Bar + @Baz + @Foo(anns = {@Bar("b"), @Bar("a")}) + A sortedWithNestedBar(); + + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Foo(ints = {1, 2}) + @Foo({"b"}) + A sortedRepeatableAnnotation(); + // BUG: Diagnostic matches: X + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Bar + @Foo(ints = {1, 2}) + A unsortedRepeatableAnnotation(); + } + + """) .doTest(); } @@ -115,128 +118,134 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import java.lang.annotation.Repeatable;", - "", - "interface A {", - " @Repeatable(Foos.class)", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] ints() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @interface Baz {", - " String[] str() default {};", - " }", - "", - " @interface Foos {", - " Foo[] value();", - " }", - "", - " @Bar", - " A singleAnnotation();", - "", - " @Bar", - " @Foo", - " A sortedAnnotations();", - "", - " @Foo", - " @Bar", - " A unsortedAnnotations();", - "", - " @Foo()", - " @Baz()", - " @Bar", - " A unsortedAnnotationsWithSomeParens();", - "", - " @Bar", - " @Baz(str = {\"a\", \"b\"})", - " @Foo()", - " A unsortedAnnotationsOneContainingAttributes();", - "", - " @Baz(str = {\"a\", \"b\"})", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Bar({\"b\"})", - " A unsortedAnnotationsWithAttributes();", - "", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Foo(ints = {1, 2})", - " @Foo({\"b\"})", - " A sortedRepeatableAnnotation();", - "", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Bar", - " @Foo(ints = {1, 2})", - " A unsortedRepeatableAnnotation();", - "}") + """ + import java.lang.annotation.Repeatable; + + interface A { + @Repeatable(Foos.class) + @interface Foo { + String[] value() default {}; + + int[] ints() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @interface Baz { + String[] str() default {}; + } + + @interface Foos { + Foo[] value(); + } + + @Bar + A singleAnnotation(); + + @Bar + @Foo + A sortedAnnotations(); + + @Foo + @Bar + A unsortedAnnotations(); + + @Foo() + @Baz() + @Bar + A unsortedAnnotationsWithSomeParens(); + + @Bar + @Baz(str = {"a", "b"}) + @Foo() + A unsortedAnnotationsOneContainingAttributes(); + + @Baz(str = {"a", "b"}) + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Bar({"b"}) + A unsortedAnnotationsWithAttributes(); + + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Foo(ints = {1, 2}) + @Foo({"b"}) + A sortedRepeatableAnnotation(); + + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Bar + @Foo(ints = {1, 2}) + A unsortedRepeatableAnnotation(); + } + + """) .addOutputLines( "out/A.java", - "import java.lang.annotation.Repeatable;", - "", - "interface A {", - " @Repeatable(Foos.class)", - " @interface Foo {", - " String[] value() default {};", - "", - " int[] ints() default {};", - "", - " Bar[] anns() default {};", - " }", - "", - " @interface Bar {", - " String[] value() default {};", - " }", - "", - " @interface Baz {", - " String[] str() default {};", - " }", - "", - " @interface Foos {", - " Foo[] value();", - " }", - "", - " @Bar", - " A singleAnnotation();", - "", - " @Bar", - " @Foo", - " A sortedAnnotations();", - "", - " @Bar", - " @Foo", - " A unsortedAnnotations();", - "", - " @Bar", - " @Baz()", - " @Foo()", - " A unsortedAnnotationsWithSomeParens();", - "", - " @Bar", - " @Baz(str = {\"a\", \"b\"})", - " @Foo()", - " A unsortedAnnotationsOneContainingAttributes();", - "", - " @Bar({\"b\"})", - " @Baz(str = {\"a\", \"b\"})", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " A unsortedAnnotationsWithAttributes();", - "", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Foo(ints = {1, 2})", - " @Foo({\"b\"})", - " A sortedRepeatableAnnotation();", - "", - " @Bar", - " @Foo(anns = {@Bar(\"b\"), @Bar(\"a\")})", - " @Foo(ints = {1, 2})", - " A unsortedRepeatableAnnotation();", - "}") + """ + import java.lang.annotation.Repeatable; + + interface A { + @Repeatable(Foos.class) + @interface Foo { + String[] value() default {}; + + int[] ints() default {}; + + Bar[] anns() default {}; + } + + @interface Bar { + String[] value() default {}; + } + + @interface Baz { + String[] str() default {}; + } + + @interface Foos { + Foo[] value(); + } + + @Bar + A singleAnnotation(); + + @Bar + @Foo + A sortedAnnotations(); + + @Bar + @Foo + A unsortedAnnotations(); + + @Bar + @Baz() + @Foo() + A unsortedAnnotationsWithSomeParens(); + + @Bar + @Baz(str = {"a", "b"}) + @Foo() + A unsortedAnnotationsOneContainingAttributes(); + + @Bar({"b"}) + @Baz(str = {"a", "b"}) + @Foo(anns = {@Bar("b"), @Bar("a")}) + A unsortedAnnotationsWithAttributes(); + + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Foo(ints = {1, 2}) + @Foo({"b"}) + A sortedRepeatableAnnotation(); + + @Bar + @Foo(anns = {@Bar("b"), @Bar("a")}) + @Foo(ints = {1, 2}) + A unsortedRepeatableAnnotation(); + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MethodReferenceUsageTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MethodReferenceUsageTest.java index dce5a4db134..2f786c260d2 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MethodReferenceUsageTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MethodReferenceUsageTest.java @@ -16,308 +16,311 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.google.common.collect.Streams;", - "import java.util.HashMap;", - "import java.util.Map;", - "import java.util.function.IntConsumer;", - "import java.util.function.IntFunction;", - "import java.util.stream.Stream;", - "", - "class A {", - " private final Stream s = Stream.of(1);", - " private final Map m = new HashMap<>();", - " private final Runnable thrower =", - " () -> {", - " throw new RuntimeException();", - " };", - "", - " void unaryExternalStaticFunctionCalls() {", - " s.forEach(String::valueOf);", - " // BUG: Diagnostic contains:", - " s.forEach(v -> String.valueOf(v));", - " s.forEach(", - " // BUG: Diagnostic contains:", - " (v) -> {", - " String.valueOf(v);", - " });", - " s.forEach(", - " // BUG: Diagnostic contains:", - " (Integer v) -> {", - " {", - " String.valueOf(v);", - " }", - " });", - " s.forEach(", - " v -> {", - " String.valueOf(v);", - " String.valueOf(v);", - " });", - "", - " s.map(String::valueOf);", - " // BUG: Diagnostic contains:", - " s.map(v -> String.valueOf(v));", - " // BUG: Diagnostic contains:", - " s.map((v) -> (String.valueOf(v)));", - " s.map(", - " // BUG: Diagnostic contains:", - " (Integer v) -> {", - " return String.valueOf(v);", - " });", - " s.map(", - " // BUG: Diagnostic contains:", - " (final Integer v) -> {", - " return (String.valueOf(v));", - " });", - " s.map(", - " v -> {", - " String.valueOf(v);", - " return String.valueOf(v);", - " });", - "", - " s.findFirst().orElseGet(() -> Integer.valueOf(\"0\"));", - " m.forEach((k, v) -> String.valueOf(v));", - " m.forEach((k, v) -> String.valueOf(k));", - " }", - "", - " void binaryExternalInstanceFunctionCalls() {", - " m.forEach(m::put);", - " // BUG: Diagnostic contains:", - " m.forEach((k, v) -> m.put(k, v));", - " m.forEach((k, v) -> m.put(v, k));", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (Integer k, Integer v) -> {", - " m.put(k, v);", - " });", - " m.forEach(", - " (k, v) -> {", - " m.put(k, k);", - " });", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (final Integer k, final Integer v) -> {", - " {", - " m.put(k, v);", - " }", - " });", - " m.forEach(", - " (k, v) -> {", - " {", - " m.put(v, v);", - " }", - " });", - " m.forEach((k, v) -> new HashMap().put(k, v));", - " m.forEach(", - " (k, v) -> {", - " m.put(k, v);", - " m.put(k, v);", - " });", - "", - " Streams.zip(s, s, m::put);", - " // BUG: Diagnostic contains:", - " Streams.zip(s, s, (a, b) -> m.put(a, b));", - " Streams.zip(s, s, (a, b) -> m.put(b, a));", - " // BUG: Diagnostic contains:", - " Streams.zip(s, s, (Integer a, Integer b) -> (m.put(a, b)));", - " Streams.zip(s, s, (a, b) -> (m.put(a, a)));", - " Streams.zip(", - " s,", - " s,", - " // BUG: Diagnostic contains:", - " (final Integer a, final Integer b) -> {", - " return m.put(a, b);", - " });", - " Streams.zip(", - " s,", - " s,", - " (a, b) -> {", - " return m.put(b, b);", - " });", - " Streams.zip(", - " s,", - " s,", - " // BUG: Diagnostic contains:", - " (a, b) -> {", - " return (m.put(a, b));", - " });", - " Streams.zip(", - " s,", - " s,", - " (a, b) -> {", - " return (m.put(b, a));", - " });", - " Streams.zip(", - " s,", - " s,", - " (a, b) -> {", - " m.put(a, b);", - " return m.put(a, b);", - " });", - " }", - "", - " void nullaryExternalInstanceFunctionCalls() {", - " s.map(Integer::doubleValue);", - " // BUG: Diagnostic contains:", - " s.map(i -> i.doubleValue());", - " s.map(i -> i.toString());", - " s.map(i -> s.toString());", - "", - " // BUG: Diagnostic contains:", - " Stream.of(int.class).filter(c -> c.isEnum());", - " Stream.of((Class) int.class).filter(Class::isEnum);", - " // BUG: Diagnostic contains:", - " Stream.of((Class) int.class).filter(c -> c.isEnum());", - " }", - "", - " void localFunctionCalls() {", - " s.forEach(v -> ivoid0());", - " s.forEach(v -> iint0());", - " s.forEach(v -> svoid0());", - " s.forEach(v -> sint0());", - "", - " s.forEach(this::ivoid1);", - " // BUG: Diagnostic contains:", - " s.forEach(v -> ivoid1(v));", - " s.forEach(", - " // BUG: Diagnostic contains:", - " v -> {", - " ivoid1(v);", - " });", - " s.forEach(this::iint1);", - " // BUG: Diagnostic contains:", - " s.forEach(v -> iint1(v));", - " s.forEach(", - " // BUG: Diagnostic contains:", - " v -> {", - " iint1(v);", - " });", - "", - " s.forEach(A::svoid1);", - " // BUG: Diagnostic contains:", - " s.forEach(v -> svoid1(v));", - " s.forEach(", - " // BUG: Diagnostic contains:", - " v -> {", - " svoid1(v);", - " });", - " s.forEach(A::sint1);", - " // BUG: Diagnostic contains:", - " s.forEach(v -> sint1(v));", - " s.forEach(", - " // BUG: Diagnostic contains:", - " v -> {", - " sint1(v);", - " });", - "", - " s.forEach(v -> ivoid2(v, v));", - " s.forEach(v -> iint2(v, v));", - " s.forEach(v -> svoid2(v, v));", - " s.forEach(v -> sint2(v, v));", - "", - " m.forEach((k, v) -> ivoid0());", - " m.forEach((k, v) -> iint0());", - " m.forEach((k, v) -> svoid0());", - " m.forEach((k, v) -> sint0());", - "", - " m.forEach(this::ivoid2);", - " // BUG: Diagnostic contains:", - " m.forEach((k, v) -> ivoid2(k, v));", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (k, v) -> {", - " ivoid2(k, v);", - " });", - " m.forEach(this::iint2);", - " // BUG: Diagnostic contains:", - " m.forEach((k, v) -> iint2(k, v));", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (k, v) -> {", - " iint2(k, v);", - " });", - "", - " m.forEach(A::svoid2);", - " // BUG: Diagnostic contains:", - " m.forEach((k, v) -> svoid2(k, v));", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (k, v) -> {", - " svoid2(k, v);", - " });", - " m.forEach(A::sint2);", - " // BUG: Diagnostic contains:", - " m.forEach((k, v) -> sint2(k, v));", - " m.forEach(", - " // BUG: Diagnostic contains:", - " (k, v) -> {", - " sint2(k, v);", - " });", - " }", - "", - " void functionCallsWhoseReplacementWouldBeAmbiguous() {", - " receiver(", - " i -> {", - " Integer.toString(i);", - " });", - " }", - "", - " void assortedOtherEdgeCases() {", - " s.forEach(v -> String.valueOf(v.toString()));", - " TernaryOp o1 = (a, b, c) -> String.valueOf(a);", - " TernaryOp o2 = (a, b, c) -> String.valueOf(b);", - " TernaryOp o3 = (a, b, c) -> String.valueOf(c);", - " TernaryOp o4 = (a, b, c) -> c.concat(a);", - " TernaryOp o5 = (a, b, c) -> c.concat(b);", - " TernaryOp o6 = (a, b, c) -> a.concat(c);", - " TernaryOp o7 = (a, b, c) -> b.concat(c);", - " }", - "", - " void receiver(IntFunction op) {}", - "", - " void receiver(IntConsumer op) {}", - "", - " void ivoid0() {}", - "", - " void ivoid1(int a) {}", - "", - " void ivoid2(int a, int b) {}", - "", - " int iint0() {", - " return 0;", - " }", - "", - " int iint1(int a) {", - " return 0;", - " }", - "", - " int iint2(int a, int b) {", - " return 0;", - " }", - "", - " static void svoid0() {}", - "", - " static void svoid1(int a) {}", - "", - " static void svoid2(int a, int b) {}", - "", - " static void svoid3(int a, int b, int c) {}", - "", - " static int sint0() {", - " return 0;", - " }", - "", - " static int sint1(int a) {", - " return 0;", - " }", - "", - " static int sint2(int a, int b) {", - " return 0;", - " }", - "", - " interface TernaryOp {", - " String collect(String a, String b, String c);", - " }", - "}") + """ + import com.google.common.collect.Streams; + import java.util.HashMap; + import java.util.Map; + import java.util.function.IntConsumer; + import java.util.function.IntFunction; + import java.util.stream.Stream; + + class A { + private final Stream s = Stream.of(1); + private final Map m = new HashMap<>(); + private final Runnable thrower = + () -> { + throw new RuntimeException(); + }; + + void unaryExternalStaticFunctionCalls() { + s.forEach(String::valueOf); + // BUG: Diagnostic contains: + s.forEach(v -> String.valueOf(v)); + s.forEach( + // BUG: Diagnostic contains: + (v) -> { + String.valueOf(v); + }); + s.forEach( + // BUG: Diagnostic contains: + (Integer v) -> { + { + String.valueOf(v); + } + }); + s.forEach( + v -> { + String.valueOf(v); + String.valueOf(v); + }); + + s.map(String::valueOf); + // BUG: Diagnostic contains: + s.map(v -> String.valueOf(v)); + // BUG: Diagnostic contains: + s.map((v) -> (String.valueOf(v))); + s.map( + // BUG: Diagnostic contains: + (Integer v) -> { + return String.valueOf(v); + }); + s.map( + // BUG: Diagnostic contains: + (final Integer v) -> { + return (String.valueOf(v)); + }); + s.map( + v -> { + String.valueOf(v); + return String.valueOf(v); + }); + + s.findFirst().orElseGet(() -> Integer.valueOf("0")); + m.forEach((k, v) -> String.valueOf(v)); + m.forEach((k, v) -> String.valueOf(k)); + } + + void binaryExternalInstanceFunctionCalls() { + m.forEach(m::put); + // BUG: Diagnostic contains: + m.forEach((k, v) -> m.put(k, v)); + m.forEach((k, v) -> m.put(v, k)); + m.forEach( + // BUG: Diagnostic contains: + (Integer k, Integer v) -> { + m.put(k, v); + }); + m.forEach( + (k, v) -> { + m.put(k, k); + }); + m.forEach( + // BUG: Diagnostic contains: + (final Integer k, final Integer v) -> { + { + m.put(k, v); + } + }); + m.forEach( + (k, v) -> { + { + m.put(v, v); + } + }); + m.forEach((k, v) -> new HashMap().put(k, v)); + m.forEach( + (k, v) -> { + m.put(k, v); + m.put(k, v); + }); + + Streams.zip(s, s, m::put); + // BUG: Diagnostic contains: + Streams.zip(s, s, (a, b) -> m.put(a, b)); + Streams.zip(s, s, (a, b) -> m.put(b, a)); + // BUG: Diagnostic contains: + Streams.zip(s, s, (Integer a, Integer b) -> (m.put(a, b))); + Streams.zip(s, s, (a, b) -> (m.put(a, a))); + Streams.zip( + s, + s, + // BUG: Diagnostic contains: + (final Integer a, final Integer b) -> { + return m.put(a, b); + }); + Streams.zip( + s, + s, + (a, b) -> { + return m.put(b, b); + }); + Streams.zip( + s, + s, + // BUG: Diagnostic contains: + (a, b) -> { + return (m.put(a, b)); + }); + Streams.zip( + s, + s, + (a, b) -> { + return (m.put(b, a)); + }); + Streams.zip( + s, + s, + (a, b) -> { + m.put(a, b); + return m.put(a, b); + }); + } + + void nullaryExternalInstanceFunctionCalls() { + s.map(Integer::doubleValue); + // BUG: Diagnostic contains: + s.map(i -> i.doubleValue()); + s.map(i -> i.toString()); + s.map(i -> s.toString()); + + // BUG: Diagnostic contains: + Stream.of(int.class).filter(c -> c.isEnum()); + Stream.of((Class) int.class).filter(Class::isEnum); + // BUG: Diagnostic contains: + Stream.of((Class) int.class).filter(c -> c.isEnum()); + } + + void localFunctionCalls() { + s.forEach(v -> ivoid0()); + s.forEach(v -> iint0()); + s.forEach(v -> svoid0()); + s.forEach(v -> sint0()); + + s.forEach(this::ivoid1); + // BUG: Diagnostic contains: + s.forEach(v -> ivoid1(v)); + s.forEach( + // BUG: Diagnostic contains: + v -> { + ivoid1(v); + }); + s.forEach(this::iint1); + // BUG: Diagnostic contains: + s.forEach(v -> iint1(v)); + s.forEach( + // BUG: Diagnostic contains: + v -> { + iint1(v); + }); + + s.forEach(A::svoid1); + // BUG: Diagnostic contains: + s.forEach(v -> svoid1(v)); + s.forEach( + // BUG: Diagnostic contains: + v -> { + svoid1(v); + }); + s.forEach(A::sint1); + // BUG: Diagnostic contains: + s.forEach(v -> sint1(v)); + s.forEach( + // BUG: Diagnostic contains: + v -> { + sint1(v); + }); + + s.forEach(v -> ivoid2(v, v)); + s.forEach(v -> iint2(v, v)); + s.forEach(v -> svoid2(v, v)); + s.forEach(v -> sint2(v, v)); + + m.forEach((k, v) -> ivoid0()); + m.forEach((k, v) -> iint0()); + m.forEach((k, v) -> svoid0()); + m.forEach((k, v) -> sint0()); + + m.forEach(this::ivoid2); + // BUG: Diagnostic contains: + m.forEach((k, v) -> ivoid2(k, v)); + m.forEach( + // BUG: Diagnostic contains: + (k, v) -> { + ivoid2(k, v); + }); + m.forEach(this::iint2); + // BUG: Diagnostic contains: + m.forEach((k, v) -> iint2(k, v)); + m.forEach( + // BUG: Diagnostic contains: + (k, v) -> { + iint2(k, v); + }); + + m.forEach(A::svoid2); + // BUG: Diagnostic contains: + m.forEach((k, v) -> svoid2(k, v)); + m.forEach( + // BUG: Diagnostic contains: + (k, v) -> { + svoid2(k, v); + }); + m.forEach(A::sint2); + // BUG: Diagnostic contains: + m.forEach((k, v) -> sint2(k, v)); + m.forEach( + // BUG: Diagnostic contains: + (k, v) -> { + sint2(k, v); + }); + } + + void functionCallsWhoseReplacementWouldBeAmbiguous() { + receiver( + i -> { + Integer.toString(i); + }); + } + + void assortedOtherEdgeCases() { + s.forEach(v -> String.valueOf(v.toString())); + TernaryOp o1 = (a, b, c) -> String.valueOf(a); + TernaryOp o2 = (a, b, c) -> String.valueOf(b); + TernaryOp o3 = (a, b, c) -> String.valueOf(c); + TernaryOp o4 = (a, b, c) -> c.concat(a); + TernaryOp o5 = (a, b, c) -> c.concat(b); + TernaryOp o6 = (a, b, c) -> a.concat(c); + TernaryOp o7 = (a, b, c) -> b.concat(c); + } + + void receiver(IntFunction op) {} + + void receiver(IntConsumer op) {} + + void ivoid0() {} + + void ivoid1(int a) {} + + void ivoid2(int a, int b) {} + + int iint0() { + return 0; + } + + int iint1(int a) { + return 0; + } + + int iint2(int a, int b) { + return 0; + } + + static void svoid0() {} + + static void svoid1(int a) {} + + static void svoid2(int a, int b) {} + + static void svoid3(int a, int b, int c) {} + + static int sint0() { + return 0; + } + + static int sint1(int a) { + return 0; + } + + static int sint2(int a, int b) { + return 0; + } + + interface TernaryOp { + String collect(String a, String b, String c); + } + } + + """) .doTest(); } @@ -326,111 +329,117 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static java.util.Collections.emptyList;", - "", - "import java.util.Collections;", - "import java.util.List;", - "import java.util.Map;", - "import java.util.function.IntSupplier;", - "import java.util.function.Supplier;", - "import java.util.stream.Stream;", - "", - "class A {", - " static class B extends A {", - " final A a = new B();", - " final B b = new B();", - "", - " IntSupplier intSup;", - " Supplier> listSup;", - "", - " void m() {", - " intSup = () -> a.iint0();", - " intSup = () -> b.iint0();", - " intSup = () -> this.iint0();", - " intSup = () -> super.iint0();", - "", - " intSup = () -> a.sint0();", - " intSup = () -> b.sint0();", - " intSup = () -> this.sint0();", - " intSup = () -> super.sint0();", - " intSup = () -> A.sint0();", - " intSup = () -> B.sint0();", - "", - " listSup = () -> Collections.emptyList();", - " listSup = () -> emptyList();", - "", - " Stream.of((Class) int.class).filter(c -> c.isEnum());", - " Stream.of((Map) null).map(Map::keySet).map(s -> s.size());", - " }", - "", - " @Override", - " int iint0() {", - " return 0;", - " }", - " }", - "", - " int iint0() {", - " return 0;", - " }", - "", - " static int sint0() {", - " return 0;", - " }", - "}") + """ + import static java.util.Collections.emptyList; + + import java.util.Collections; + import java.util.List; + import java.util.Map; + import java.util.function.IntSupplier; + import java.util.function.Supplier; + import java.util.stream.Stream; + + class A { + static class B extends A { + final A a = new B(); + final B b = new B(); + + IntSupplier intSup; + Supplier> listSup; + + void m() { + intSup = () -> a.iint0(); + intSup = () -> b.iint0(); + intSup = () -> this.iint0(); + intSup = () -> super.iint0(); + + intSup = () -> a.sint0(); + intSup = () -> b.sint0(); + intSup = () -> this.sint0(); + intSup = () -> super.sint0(); + intSup = () -> A.sint0(); + intSup = () -> B.sint0(); + + listSup = () -> Collections.emptyList(); + listSup = () -> emptyList(); + + Stream.of((Class) int.class).filter(c -> c.isEnum()); + Stream.of((Map) null).map(Map::keySet).map(s -> s.size()); + } + + @Override + int iint0() { + return 0; + } + } + + int iint0() { + return 0; + } + + static int sint0() { + return 0; + } + } + + """) .addOutputLines( "out/A.java", - "import static java.util.Collections.emptyList;", - "", - "import java.util.Collections;", - "import java.util.List;", - "import java.util.Map;", - "import java.util.Set;", - "import java.util.function.IntSupplier;", - "import java.util.function.Supplier;", - "import java.util.stream.Stream;", - "", - "class A {", - " static class B extends A {", - " final A a = new B();", - " final B b = new B();", - "", - " IntSupplier intSup;", - " Supplier> listSup;", - "", - " void m() {", - " intSup = a::iint0;", - " intSup = b::iint0;", - " intSup = this::iint0;", - " intSup = super::iint0;", - "", - " intSup = () -> a.sint0();", - " intSup = () -> b.sint0();", - " intSup = () -> this.sint0();", - " intSup = () -> super.sint0();", - " intSup = A::sint0;", - " intSup = B::sint0;", - "", - " listSup = Collections::emptyList;", - " listSup = Collections::emptyList;", - "", - " Stream.of((Class) int.class).filter(Class::isEnum);", - " Stream.of((Map) null).map(Map::keySet).map(Set::size);", - " }", - "", - " @Override", - " int iint0() {", - " return 0;", - " }", - " }", - "", - " int iint0() {", - " return 0;", - " }", - "", - " static int sint0() {", - " return 0;", - " }", - "}") + """ + import static java.util.Collections.emptyList; + + import java.util.Collections; + import java.util.List; + import java.util.Map; + import java.util.Set; + import java.util.function.IntSupplier; + import java.util.function.Supplier; + import java.util.stream.Stream; + + class A { + static class B extends A { + final A a = new B(); + final B b = new B(); + + IntSupplier intSup; + Supplier> listSup; + + void m() { + intSup = a::iint0; + intSup = b::iint0; + intSup = this::iint0; + intSup = super::iint0; + + intSup = () -> a.sint0(); + intSup = () -> b.sint0(); + intSup = () -> this.sint0(); + intSup = () -> super.sint0(); + intSup = A::sint0; + intSup = B::sint0; + + listSup = Collections::emptyList; + listSup = Collections::emptyList; + + Stream.of((Class) int.class).filter(Class::isEnum); + Stream.of((Map) null).map(Map::keySet).map(Set::size); + } + + @Override + int iint0() { + return 0; + } + } + + int iint0() { + return 0; + } + + static int sint0() { + return 0; + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MissingRefasterAnnotationTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MissingRefasterAnnotationTest.java index 3611a2f6f8f..e5dadc02c9f 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MissingRefasterAnnotationTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MissingRefasterAnnotationTest.java @@ -18,76 +18,79 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.google.errorprone.refaster.annotation.AfterTemplate;", - "import com.google.errorprone.refaster.annotation.AlsoNegation;", - "import com.google.errorprone.refaster.annotation.BeforeTemplate;", - "import java.util.Map;", - "", - "class A {", - " // BUG: Diagnostic matches: X", - " static final class MethodLacksBeforeTemplateAnnotation {", - " @BeforeTemplate", - " boolean before1(String string) {", - " return string.equals(\"\");", - " }", - "", - " // @BeforeTemplate is missing", - " boolean before2(String string) {", - " return string.length() == 0;", - " }", - "", - " @AfterTemplate", - " @AlsoNegation", - " boolean after(String string) {", - " return string.isEmpty();", - " }", - " }", - "", - " // BUG: Diagnostic matches: X", - " static final class MethodLacksAfterTemplateAnnotation {", - " @BeforeTemplate", - " boolean before(String string) {", - " return string.equals(\"\");", - " }", - "", - " // @AfterTemplate is missing", - " boolean after(String string) {", - " return string.isEmpty();", - " }", - " }", - "", - " // BUG: Diagnostic matches: X", - " abstract class MethodLacksPlaceholderAnnotation {", - " // @Placeholder is missing", - " abstract V function(K key);", - "", - " @BeforeTemplate", - " void before(Map map, K key) {", - " if (!map.containsKey(key)) {", - " map.put(key, function(key));", - " }", - " }", - "", - " @AfterTemplate", - " void after(Map map, K key) {", - " map.computeIfAbsent(key, k -> function(k));", - " }", - " }", - "", - " static final class ValidRefasterTemplate {", - " @BeforeTemplate", - " void unusedPureFunctionCall(Object o) {", - " o.toString();", - " }", - " }", - "", - " static final class NotARefasterTemplate {", - " @Override", - " public String toString() {", - " return \"This is not a Refaster template\";", - " }", - " }", - "}") + """ + import com.google.errorprone.refaster.annotation.AfterTemplate; + import com.google.errorprone.refaster.annotation.AlsoNegation; + import com.google.errorprone.refaster.annotation.BeforeTemplate; + import java.util.Map; + + class A { + // BUG: Diagnostic matches: X + static final class MethodLacksBeforeTemplateAnnotation { + @BeforeTemplate + boolean before1(String string) { + return string.equals(""); + } + + // @BeforeTemplate is missing + boolean before2(String string) { + return string.length() == 0; + } + + @AfterTemplate + @AlsoNegation + boolean after(String string) { + return string.isEmpty(); + } + } + + // BUG: Diagnostic matches: X + static final class MethodLacksAfterTemplateAnnotation { + @BeforeTemplate + boolean before(String string) { + return string.equals(""); + } + + // @AfterTemplate is missing + boolean after(String string) { + return string.isEmpty(); + } + } + + // BUG: Diagnostic matches: X + abstract class MethodLacksPlaceholderAnnotation { + // @Placeholder is missing + abstract V function(K key); + + @BeforeTemplate + void before(Map map, K key) { + if (!map.containsKey(key)) { + map.put(key, function(key)); + } + } + + @AfterTemplate + void after(Map map, K key) { + map.computeIfAbsent(key, k -> function(k)); + } + } + + static final class ValidRefasterTemplate { + @BeforeTemplate + void unusedPureFunctionCall(Object o) { + o.toString(); + } + } + + static final class NotARefasterTemplate { + @Override + public String toString() { + return "This is not a Refaster template"; + } + } + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MockitoStubbingTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MockitoStubbingTest.java index 50bb0414540..117de78557e 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MockitoStubbingTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/MockitoStubbingTest.java @@ -16,40 +16,43 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static org.mockito.ArgumentMatchers.eq;", - "import static org.mockito.ArgumentMatchers.notNull;", - "import static org.mockito.Mockito.doAnswer;", - "import static org.mockito.Mockito.mock;", - "", - "import java.util.function.BiConsumer;", - "import java.util.function.Consumer;", - "import org.mockito.ArgumentMatchers;", - "", - "class A {", - " void m() {", - " Runnable runnable = mock(Runnable.class);", - " doAnswer(inv -> null).when(runnable).run();", - "", - " Consumer consumer = mock(Consumer.class);", - " doAnswer(inv -> null).when(consumer).accept(\"foo\");", - " doAnswer(inv -> null).when(consumer).accept(notNull());", - " // BUG: Diagnostic contains:", - " doAnswer(inv -> null).when(consumer).accept(ArgumentMatchers.eq(\"foo\"));", - " // BUG: Diagnostic contains:", - " doAnswer(inv -> null).when(consumer).accept(eq(toString()));", - "", - " BiConsumer biConsumer = mock(BiConsumer.class);", - " doAnswer(inv -> null).when(biConsumer).accept(0, \"foo\");", - " doAnswer(inv -> null).when(biConsumer).accept(eq(0), notNull());", - " doAnswer(inv -> null).when(biConsumer).accept(notNull(), eq(\"foo\"));", - " doAnswer(inv -> null)", - " .when(biConsumer)", - " // BUG: Diagnostic contains:", - " .accept(ArgumentMatchers.eq(0), ArgumentMatchers.eq(\"foo\"));", - " // BUG: Diagnostic contains:", - " doAnswer(inv -> null).when(biConsumer).accept(eq(hashCode()), eq(toString()));", - " }", - "}") + """ + import static org.mockito.ArgumentMatchers.eq; + import static org.mockito.ArgumentMatchers.notNull; + import static org.mockito.Mockito.doAnswer; + import static org.mockito.Mockito.mock; + + import java.util.function.BiConsumer; + import java.util.function.Consumer; + import org.mockito.ArgumentMatchers; + + class A { + void m() { + Runnable runnable = mock(Runnable.class); + doAnswer(inv -> null).when(runnable).run(); + + Consumer consumer = mock(Consumer.class); + doAnswer(inv -> null).when(consumer).accept("foo"); + doAnswer(inv -> null).when(consumer).accept(notNull()); + // BUG: Diagnostic contains: + doAnswer(inv -> null).when(consumer).accept(ArgumentMatchers.eq("foo")); + // BUG: Diagnostic contains: + doAnswer(inv -> null).when(consumer).accept(eq(toString())); + + BiConsumer biConsumer = mock(BiConsumer.class); + doAnswer(inv -> null).when(biConsumer).accept(0, "foo"); + doAnswer(inv -> null).when(biConsumer).accept(eq(0), notNull()); + doAnswer(inv -> null).when(biConsumer).accept(notNull(), eq("foo")); + doAnswer(inv -> null) + .when(biConsumer) + // BUG: Diagnostic contains: + .accept(ArgumentMatchers.eq(0), ArgumentMatchers.eq("foo")); + // BUG: Diagnostic contains: + doAnswer(inv -> null).when(biConsumer).accept(eq(hashCode()), eq(toString())); + } + } + + """) .doTest(); } @@ -58,48 +61,54 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static org.mockito.ArgumentMatchers.eq;", - "import static org.mockito.Mockito.doAnswer;", - "import static org.mockito.Mockito.mock;", - "", - "import java.util.function.BiConsumer;", - "import java.util.function.Consumer;", - "import org.mockito.ArgumentMatchers;", - "", - "class A {", - " void m() {", - " Consumer consumer = mock(Consumer.class);", - " doAnswer(inv -> null).when(consumer).accept(ArgumentMatchers.eq(\"foo\"));", - " doAnswer(inv -> null).when(consumer).accept(eq(toString()));", - "", - " BiConsumer biConsumer = mock(BiConsumer.class);", - " doAnswer(inv -> null)", - " .when(biConsumer)", - " .accept(ArgumentMatchers.eq(0), ArgumentMatchers.eq(\"foo\"));", - " doAnswer(inv -> null).when(biConsumer).accept(eq(hashCode()), eq(toString()));", - " }", - "}") + """ + import static org.mockito.ArgumentMatchers.eq; + import static org.mockito.Mockito.doAnswer; + import static org.mockito.Mockito.mock; + + import java.util.function.BiConsumer; + import java.util.function.Consumer; + import org.mockito.ArgumentMatchers; + + class A { + void m() { + Consumer consumer = mock(Consumer.class); + doAnswer(inv -> null).when(consumer).accept(ArgumentMatchers.eq("foo")); + doAnswer(inv -> null).when(consumer).accept(eq(toString())); + + BiConsumer biConsumer = mock(BiConsumer.class); + doAnswer(inv -> null) + .when(biConsumer) + .accept(ArgumentMatchers.eq(0), ArgumentMatchers.eq("foo")); + doAnswer(inv -> null).when(biConsumer).accept(eq(hashCode()), eq(toString())); + } + } + + """) .addOutputLines( "out/A.java", - "import static org.mockito.ArgumentMatchers.eq;", - "import static org.mockito.Mockito.doAnswer;", - "import static org.mockito.Mockito.mock;", - "", - "import java.util.function.BiConsumer;", - "import java.util.function.Consumer;", - "import org.mockito.ArgumentMatchers;", - "", - "class A {", - " void m() {", - " Consumer consumer = mock(Consumer.class);", - " doAnswer(inv -> null).when(consumer).accept(\"foo\");", - " doAnswer(inv -> null).when(consumer).accept(toString());", - "", - " BiConsumer biConsumer = mock(BiConsumer.class);", - " doAnswer(inv -> null).when(biConsumer).accept(0, \"foo\");", - " doAnswer(inv -> null).when(biConsumer).accept(hashCode(), toString());", - " }", - "}") + """ + import static org.mockito.ArgumentMatchers.eq; + import static org.mockito.Mockito.doAnswer; + import static org.mockito.Mockito.mock; + + import java.util.function.BiConsumer; + import java.util.function.Consumer; + import org.mockito.ArgumentMatchers; + + class A { + void m() { + Consumer consumer = mock(Consumer.class); + doAnswer(inv -> null).when(consumer).accept("foo"); + doAnswer(inv -> null).when(consumer).accept(toString()); + + BiConsumer biConsumer = mock(BiConsumer.class); + doAnswer(inv -> null).when(biConsumer).accept(0, "foo"); + doAnswer(inv -> null).when(biConsumer).accept(hashCode(), toString()); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/PrimitiveComparisonTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/PrimitiveComparisonTest.java index cace3d7a2ec..2567cfc2a02 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/PrimitiveComparisonTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/PrimitiveComparisonTest.java @@ -23,98 +23,101 @@ void byteComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "", - "class A {", - " {", - " // BUG: Diagnostic contains:", - " Comparator.comparing(this::toPrimitive);", - " Comparator.comparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " Comparator.comparing(o -> (byte) 0);", - " Comparator.comparing(o -> (byte) 0, cmp());", - " Comparator.comparing(this::toBoxed);", - " Comparator.comparing(this::toBoxed, cmp());", - " Comparator.comparing(o -> Byte.valueOf((byte) 0));", - " Comparator.comparing(o -> Byte.valueOf((byte) 0), cmp());", - " Comparator.comparing(toBoxed());", - " Comparator.comparing(toBoxed(), cmp());", - " Comparator.comparingInt(this::toPrimitive);", - " Comparator.comparingInt(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " Comparator.comparingInt(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingInt(o -> Byte.valueOf((byte) 0));", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(o -> Byte.valueOf((byte) 0));", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> Byte.valueOf((byte) 0));", - "", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(this::toPrimitive);", - " cmp().thenComparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(o -> (byte) 0);", - " cmp().thenComparing(o -> (byte) 0, cmp());", - " cmp().thenComparing(this::toBoxed);", - " cmp().thenComparing(this::toBoxed, cmp());", - " cmp().thenComparing(o -> Byte.valueOf((byte) 0));", - " cmp().thenComparing(o -> Byte.valueOf((byte) 0), cmp());", - " cmp().thenComparing(toBoxed());", - " cmp().thenComparing(toBoxed(), cmp());", - " cmp().thenComparingInt(this::toPrimitive);", - " cmp().thenComparingInt(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingInt(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingInt(o -> Byte.valueOf((byte) 0));", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(o -> Byte.valueOf((byte) 0));", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> (byte) 0);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> Byte.valueOf((byte) 0));", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private byte toPrimitive(Object o) {", - " return 0;", - " }", - "", - " private Byte toBoxed(Object o) {", - " return 0;", - " }", - "", - " private Function toBoxed() {", - " return o -> 0;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + + class A { + { + // BUG: Diagnostic contains: + Comparator.comparing(this::toPrimitive); + Comparator.comparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + Comparator.comparing(o -> (byte) 0); + Comparator.comparing(o -> (byte) 0, cmp()); + Comparator.comparing(this::toBoxed); + Comparator.comparing(this::toBoxed, cmp()); + Comparator.comparing(o -> Byte.valueOf((byte) 0)); + Comparator.comparing(o -> Byte.valueOf((byte) 0), cmp()); + Comparator.comparing(toBoxed()); + Comparator.comparing(toBoxed(), cmp()); + Comparator.comparingInt(this::toPrimitive); + Comparator.comparingInt(o -> (byte) 0); + // BUG: Diagnostic contains: + Comparator.comparingInt(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingInt(o -> Byte.valueOf((byte) 0)); + // BUG: Diagnostic contains: + Comparator.comparingLong(this::toPrimitive); + // BUG: Diagnostic contains: + Comparator.comparingLong(o -> (byte) 0); + // BUG: Diagnostic contains: + Comparator.comparingLong(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingLong(o -> Byte.valueOf((byte) 0)); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> (byte) 0); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> Byte.valueOf((byte) 0)); + + // BUG: Diagnostic contains: + cmp().thenComparing(this::toPrimitive); + cmp().thenComparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + cmp().thenComparing(o -> (byte) 0); + cmp().thenComparing(o -> (byte) 0, cmp()); + cmp().thenComparing(this::toBoxed); + cmp().thenComparing(this::toBoxed, cmp()); + cmp().thenComparing(o -> Byte.valueOf((byte) 0)); + cmp().thenComparing(o -> Byte.valueOf((byte) 0), cmp()); + cmp().thenComparing(toBoxed()); + cmp().thenComparing(toBoxed(), cmp()); + cmp().thenComparingInt(this::toPrimitive); + cmp().thenComparingInt(o -> (byte) 0); + // BUG: Diagnostic contains: + cmp().thenComparingInt(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingInt(o -> Byte.valueOf((byte) 0)); + // BUG: Diagnostic contains: + cmp().thenComparingLong(this::toPrimitive); + // BUG: Diagnostic contains: + cmp().thenComparingLong(o -> (byte) 0); + // BUG: Diagnostic contains: + cmp().thenComparingLong(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingLong(o -> Byte.valueOf((byte) 0)); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> (byte) 0); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> Byte.valueOf((byte) 0)); + } + + private Comparator cmp() { + return null; + } + + private byte toPrimitive(Object o) { + return 0; + } + + private Byte toBoxed(Object o) { + return 0; + } + + private Function toBoxed() { + return o -> 0; + } + } + + """) .doTest(); } @@ -123,105 +126,108 @@ void intComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "import java.util.function.ToIntFunction;", - "", - "class A {", - " {", - " // BUG: Diagnostic contains:", - " Comparator.comparing(this::toPrimitive);", - " Comparator.comparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " Comparator.comparing(o -> 0);", - " Comparator.comparing(o -> 0, cmp());", - " Comparator.comparing(this::toBoxed);", - " Comparator.comparing(this::toBoxed, cmp());", - " Comparator.comparing(o -> Integer.valueOf(0));", - " Comparator.comparing(o -> Integer.valueOf(0), cmp());", - " Comparator.comparing(toBoxed());", - " Comparator.comparing(toBoxed(), cmp());", - " Comparator.comparingInt(this::toPrimitive);", - " Comparator.comparingInt(o -> 0);", - " Comparator.comparingInt(toPrimitive());", - " // BUG: Diagnostic contains:", - " Comparator.comparingInt(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingInt(o -> Integer.valueOf(0));", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(o -> 0);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(o -> Integer.valueOf(0));", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> 0);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> Integer.valueOf(0));", - "", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(this::toPrimitive);", - " cmp().thenComparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(o -> 0);", - " cmp().thenComparing(o -> 0, cmp());", - " cmp().thenComparing(this::toBoxed);", - " cmp().thenComparing(this::toBoxed, cmp());", - " cmp().thenComparing(o -> Integer.valueOf(0));", - " cmp().thenComparing(o -> Integer.valueOf(0), cmp());", - " cmp().thenComparing(toBoxed());", - " cmp().thenComparing(toBoxed(), cmp());", - " cmp().thenComparingInt(this::toPrimitive);", - " cmp().thenComparingInt(o -> 0);", - " cmp().thenComparingInt(toPrimitive());", - " // BUG: Diagnostic contains:", - " cmp().thenComparingInt(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingInt(o -> Integer.valueOf(0));", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(o -> 0);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(o -> Integer.valueOf(0));", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> 0);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> Integer.valueOf(0));", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private int toPrimitive(Object o) {", - " return 0;", - " }", - "", - " private Integer toBoxed(Object o) {", - " return 0;", - " }", - "", - " private Function toBoxed() {", - " return o -> 0;", - " }", - "", - " private ToIntFunction toPrimitive() {", - " return o -> 0;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + import java.util.function.ToIntFunction; + + class A { + { + // BUG: Diagnostic contains: + Comparator.comparing(this::toPrimitive); + Comparator.comparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + Comparator.comparing(o -> 0); + Comparator.comparing(o -> 0, cmp()); + Comparator.comparing(this::toBoxed); + Comparator.comparing(this::toBoxed, cmp()); + Comparator.comparing(o -> Integer.valueOf(0)); + Comparator.comparing(o -> Integer.valueOf(0), cmp()); + Comparator.comparing(toBoxed()); + Comparator.comparing(toBoxed(), cmp()); + Comparator.comparingInt(this::toPrimitive); + Comparator.comparingInt(o -> 0); + Comparator.comparingInt(toPrimitive()); + // BUG: Diagnostic contains: + Comparator.comparingInt(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingInt(o -> Integer.valueOf(0)); + // BUG: Diagnostic contains: + Comparator.comparingLong(this::toPrimitive); + // BUG: Diagnostic contains: + Comparator.comparingLong(o -> 0); + // BUG: Diagnostic contains: + Comparator.comparingLong(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingLong(o -> Integer.valueOf(0)); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> 0); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> Integer.valueOf(0)); + + // BUG: Diagnostic contains: + cmp().thenComparing(this::toPrimitive); + cmp().thenComparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + cmp().thenComparing(o -> 0); + cmp().thenComparing(o -> 0, cmp()); + cmp().thenComparing(this::toBoxed); + cmp().thenComparing(this::toBoxed, cmp()); + cmp().thenComparing(o -> Integer.valueOf(0)); + cmp().thenComparing(o -> Integer.valueOf(0), cmp()); + cmp().thenComparing(toBoxed()); + cmp().thenComparing(toBoxed(), cmp()); + cmp().thenComparingInt(this::toPrimitive); + cmp().thenComparingInt(o -> 0); + cmp().thenComparingInt(toPrimitive()); + // BUG: Diagnostic contains: + cmp().thenComparingInt(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingInt(o -> Integer.valueOf(0)); + // BUG: Diagnostic contains: + cmp().thenComparingLong(this::toPrimitive); + // BUG: Diagnostic contains: + cmp().thenComparingLong(o -> 0); + // BUG: Diagnostic contains: + cmp().thenComparingLong(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingLong(o -> Integer.valueOf(0)); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> 0); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> Integer.valueOf(0)); + } + + private Comparator cmp() { + return null; + } + + private int toPrimitive(Object o) { + return 0; + } + + private Integer toBoxed(Object o) { + return 0; + } + + private Function toBoxed() { + return o -> 0; + } + + private ToIntFunction toPrimitive() { + return o -> 0; + } + } + + """) .doTest(); } @@ -230,89 +236,92 @@ void longComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "import java.util.function.ToLongFunction;", - "", - "class A {", - " {", - " // BUG: Diagnostic contains:", - " Comparator.comparing(this::toPrimitive);", - " Comparator.comparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " Comparator.comparing(o -> 0L);", - " Comparator.comparing(o -> 0L, cmp());", - " Comparator.comparing(this::toBoxed);", - " Comparator.comparing(this::toBoxed, cmp());", - " Comparator.comparing(o -> Long.valueOf(0));", - " Comparator.comparing(o -> Long.valueOf(0), cmp());", - " Comparator.comparing(toBoxed());", - " Comparator.comparing(toBoxed(), cmp());", - " Comparator.comparingLong(this::toPrimitive);", - " Comparator.comparingLong(o -> 0L);", - " Comparator.comparingLong(toPrimitive());", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingLong(o -> Long.valueOf(0));", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> 0L);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> Long.valueOf(0));", - "", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(this::toPrimitive);", - " cmp().thenComparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(o -> 0L);", - " cmp().thenComparing(o -> 0L, cmp());", - " cmp().thenComparing(this::toBoxed);", - " cmp().thenComparing(this::toBoxed, cmp());", - " cmp().thenComparing(o -> Long.valueOf(0));", - " cmp().thenComparing(o -> Long.valueOf(0), cmp());", - " cmp().thenComparing(toBoxed());", - " cmp().thenComparing(toBoxed(), cmp());", - " cmp().thenComparingLong(this::toPrimitive);", - " cmp().thenComparingLong(o -> 0L);", - " cmp().thenComparingLong(toPrimitive());", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingLong(o -> Long.valueOf(0));", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toPrimitive);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> 0L);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> Long.valueOf(0));", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private long toPrimitive(Object o) {", - " return 0L;", - " }", - "", - " private Long toBoxed(Object o) {", - " return 0L;", - " }", - "", - " private Function toBoxed() {", - " return o -> 0L;", - " }", - "", - " private ToLongFunction toPrimitive() {", - " return o -> 0L;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + import java.util.function.ToLongFunction; + + class A { + { + // BUG: Diagnostic contains: + Comparator.comparing(this::toPrimitive); + Comparator.comparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + Comparator.comparing(o -> 0L); + Comparator.comparing(o -> 0L, cmp()); + Comparator.comparing(this::toBoxed); + Comparator.comparing(this::toBoxed, cmp()); + Comparator.comparing(o -> Long.valueOf(0)); + Comparator.comparing(o -> Long.valueOf(0), cmp()); + Comparator.comparing(toBoxed()); + Comparator.comparing(toBoxed(), cmp()); + Comparator.comparingLong(this::toPrimitive); + Comparator.comparingLong(o -> 0L); + Comparator.comparingLong(toPrimitive()); + // BUG: Diagnostic contains: + Comparator.comparingLong(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingLong(o -> Long.valueOf(0)); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> 0L); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> Long.valueOf(0)); + + // BUG: Diagnostic contains: + cmp().thenComparing(this::toPrimitive); + cmp().thenComparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + cmp().thenComparing(o -> 0L); + cmp().thenComparing(o -> 0L, cmp()); + cmp().thenComparing(this::toBoxed); + cmp().thenComparing(this::toBoxed, cmp()); + cmp().thenComparing(o -> Long.valueOf(0)); + cmp().thenComparing(o -> Long.valueOf(0), cmp()); + cmp().thenComparing(toBoxed()); + cmp().thenComparing(toBoxed(), cmp()); + cmp().thenComparingLong(this::toPrimitive); + cmp().thenComparingLong(o -> 0L); + cmp().thenComparingLong(toPrimitive()); + // BUG: Diagnostic contains: + cmp().thenComparingLong(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingLong(o -> Long.valueOf(0)); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toPrimitive); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> 0L); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> Long.valueOf(0)); + } + + private Comparator cmp() { + return null; + } + + private long toPrimitive(Object o) { + return 0L; + } + + private Long toBoxed(Object o) { + return 0L; + } + + private Function toBoxed() { + return o -> 0L; + } + + private ToLongFunction toPrimitive() { + return o -> 0L; + } + } + + """) .doTest(); } @@ -321,66 +330,69 @@ void floatComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "", - "class A {", - " {", - " // BUG: Diagnostic contains:", - " Comparator.comparing(this::toPrimitive);", - " Comparator.comparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " Comparator.comparing(o -> 0.0f);", - " Comparator.comparing(o -> 0.0f, cmp());", - " Comparator.comparing(this::toBoxed);", - " Comparator.comparing(this::toBoxed, cmp());", - " Comparator.comparing(o -> Float.valueOf(0));", - " Comparator.comparing(o -> Float.valueOf(0), cmp());", - " Comparator.comparing(toBoxed());", - " Comparator.comparing(toBoxed(), cmp());", - " Comparator.comparingDouble(this::toPrimitive);", - " Comparator.comparingDouble(o -> 0.0f);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> Float.valueOf(0));", - "", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(this::toPrimitive);", - " cmp().thenComparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(o -> 0.0f);", - " cmp().thenComparing(o -> 0.0f, cmp());", - " cmp().thenComparing(this::toBoxed);", - " cmp().thenComparing(this::toBoxed, cmp());", - " cmp().thenComparing(o -> Float.valueOf(0));", - " cmp().thenComparing(o -> Float.valueOf(0), cmp());", - " cmp().thenComparing(toBoxed());", - " cmp().thenComparing(toBoxed(), cmp());", - " cmp().thenComparingDouble(this::toPrimitive);", - " cmp().thenComparingDouble(o -> 0.0f);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> Float.valueOf(0));", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private float toPrimitive(Object o) {", - " return 0.0f;", - " }", - "", - " private Float toBoxed(Object o) {", - " return 0.0f;", - " }", - "", - " private Function toBoxed() {", - " return o -> 0.0f;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + + class A { + { + // BUG: Diagnostic contains: + Comparator.comparing(this::toPrimitive); + Comparator.comparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + Comparator.comparing(o -> 0.0f); + Comparator.comparing(o -> 0.0f, cmp()); + Comparator.comparing(this::toBoxed); + Comparator.comparing(this::toBoxed, cmp()); + Comparator.comparing(o -> Float.valueOf(0)); + Comparator.comparing(o -> Float.valueOf(0), cmp()); + Comparator.comparing(toBoxed()); + Comparator.comparing(toBoxed(), cmp()); + Comparator.comparingDouble(this::toPrimitive); + Comparator.comparingDouble(o -> 0.0f); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> Float.valueOf(0)); + + // BUG: Diagnostic contains: + cmp().thenComparing(this::toPrimitive); + cmp().thenComparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + cmp().thenComparing(o -> 0.0f); + cmp().thenComparing(o -> 0.0f, cmp()); + cmp().thenComparing(this::toBoxed); + cmp().thenComparing(this::toBoxed, cmp()); + cmp().thenComparing(o -> Float.valueOf(0)); + cmp().thenComparing(o -> Float.valueOf(0), cmp()); + cmp().thenComparing(toBoxed()); + cmp().thenComparing(toBoxed(), cmp()); + cmp().thenComparingDouble(this::toPrimitive); + cmp().thenComparingDouble(o -> 0.0f); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> Float.valueOf(0)); + } + + private Comparator cmp() { + return null; + } + + private float toPrimitive(Object o) { + return 0.0f; + } + + private Float toBoxed(Object o) { + return 0.0f; + } + + private Function toBoxed() { + return o -> 0.0f; + } + } + + """) .doTest(); } @@ -389,73 +401,76 @@ void doubleComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "import java.util.function.ToDoubleFunction;", - "", - "class A {", - " {", - " // BUG: Diagnostic contains:", - " Comparator.comparing(this::toPrimitive);", - " Comparator.comparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " Comparator.comparing(o -> 0.0);", - " Comparator.comparing(o -> 0.0, cmp());", - " Comparator.comparing(this::toBoxed);", - " Comparator.comparing(this::toBoxed, cmp());", - " Comparator.comparing(o -> Double.valueOf(0));", - " Comparator.comparing(o -> Double.valueOf(0), cmp());", - " Comparator.comparing(toBoxed());", - " Comparator.comparing(toBoxed(), cmp());", - " Comparator.comparingDouble(this::toPrimitive);", - " Comparator.comparingDouble(o -> 0.0);", - " Comparator.comparingDouble(toPrimitive());", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " Comparator.comparingDouble(o -> Double.valueOf(0));", - "", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(this::toPrimitive);", - " cmp().thenComparing(this::toPrimitive, cmp());", - " // BUG: Diagnostic contains:", - " cmp().thenComparing(o -> 0.0);", - " cmp().thenComparing(o -> 0.0, cmp());", - " cmp().thenComparing(this::toBoxed);", - " cmp().thenComparing(this::toBoxed, cmp());", - " cmp().thenComparing(o -> Double.valueOf(0));", - " cmp().thenComparing(o -> Double.valueOf(0), cmp());", - " cmp().thenComparing(toBoxed());", - " cmp().thenComparing(toBoxed(), cmp());", - " cmp().thenComparingDouble(this::toPrimitive);", - " cmp().thenComparingDouble(o -> 0.0);", - " cmp().thenComparingDouble(toPrimitive());", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(this::toBoxed);", - " // BUG: Diagnostic contains:", - " cmp().thenComparingDouble(o -> Double.valueOf(0));", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private double toPrimitive(Object o) {", - " return 0.0;", - " }", - "", - " private Double toBoxed(Object o) {", - " return 0.0;", - " }", - "", - " private Function toBoxed() {", - " return o -> 0.0;", - " }", - "", - " private ToDoubleFunction toPrimitive() {", - " return o -> 0.0;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + import java.util.function.ToDoubleFunction; + + class A { + { + // BUG: Diagnostic contains: + Comparator.comparing(this::toPrimitive); + Comparator.comparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + Comparator.comparing(o -> 0.0); + Comparator.comparing(o -> 0.0, cmp()); + Comparator.comparing(this::toBoxed); + Comparator.comparing(this::toBoxed, cmp()); + Comparator.comparing(o -> Double.valueOf(0)); + Comparator.comparing(o -> Double.valueOf(0), cmp()); + Comparator.comparing(toBoxed()); + Comparator.comparing(toBoxed(), cmp()); + Comparator.comparingDouble(this::toPrimitive); + Comparator.comparingDouble(o -> 0.0); + Comparator.comparingDouble(toPrimitive()); + // BUG: Diagnostic contains: + Comparator.comparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + Comparator.comparingDouble(o -> Double.valueOf(0)); + + // BUG: Diagnostic contains: + cmp().thenComparing(this::toPrimitive); + cmp().thenComparing(this::toPrimitive, cmp()); + // BUG: Diagnostic contains: + cmp().thenComparing(o -> 0.0); + cmp().thenComparing(o -> 0.0, cmp()); + cmp().thenComparing(this::toBoxed); + cmp().thenComparing(this::toBoxed, cmp()); + cmp().thenComparing(o -> Double.valueOf(0)); + cmp().thenComparing(o -> Double.valueOf(0), cmp()); + cmp().thenComparing(toBoxed()); + cmp().thenComparing(toBoxed(), cmp()); + cmp().thenComparingDouble(this::toPrimitive); + cmp().thenComparingDouble(o -> 0.0); + cmp().thenComparingDouble(toPrimitive()); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(this::toBoxed); + // BUG: Diagnostic contains: + cmp().thenComparingDouble(o -> Double.valueOf(0)); + } + + private Comparator cmp() { + return null; + } + + private double toPrimitive(Object o) { + return 0.0; + } + + private Double toBoxed(Object o) { + return 0.0; + } + + private Function toBoxed() { + return o -> 0.0; + } + + private ToDoubleFunction toPrimitive() { + return o -> 0.0; + } + } + + """) .doTest(); } @@ -464,34 +479,37 @@ void stringComparison() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Comparator;", - "import java.util.function.Function;", - "", - "class A {", - " {", - " Comparator.comparing(String::valueOf);", - " Comparator.comparing(String::valueOf, cmp());", - " Comparator.comparing(o -> String.valueOf(o));", - " Comparator.comparing(o -> String.valueOf(o), cmp());", - " Comparator.comparing(toStr());", - " Comparator.comparing(toStr(), cmp());", - "", - " cmp().thenComparing(String::valueOf);", - " cmp().thenComparing(String::valueOf, cmp());", - " cmp().thenComparing(o -> String.valueOf(o));", - " cmp().thenComparing(o -> String.valueOf(o), cmp());", - " cmp().thenComparing(toStr());", - " cmp().thenComparing(toStr(), cmp());", - " }", - "", - " private Comparator cmp() {", - " return null;", - " }", - "", - " private Function toStr() {", - " return String::valueOf;", - " }", - "}") + """ + import java.util.Comparator; + import java.util.function.Function; + + class A { + { + Comparator.comparing(String::valueOf); + Comparator.comparing(String::valueOf, cmp()); + Comparator.comparing(o -> String.valueOf(o)); + Comparator.comparing(o -> String.valueOf(o), cmp()); + Comparator.comparing(toStr()); + Comparator.comparing(toStr(), cmp()); + + cmp().thenComparing(String::valueOf); + cmp().thenComparing(String::valueOf, cmp()); + cmp().thenComparing(o -> String.valueOf(o)); + cmp().thenComparing(o -> String.valueOf(o), cmp()); + cmp().thenComparing(toStr()); + cmp().thenComparing(toStr(), cmp()); + } + + private Comparator cmp() { + return null; + } + + private Function toStr() { + return String::valueOf; + } + } + + """) .doTest(); } @@ -500,78 +518,84 @@ void replacementWithPrimitiveVariants() { refactoringTestHelper .addInputLines( "in/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparing(o -> (byte) 0);", - " Comparator bCmp2 = Comparator.comparing(o -> (byte) 0);", - " Comparator cCmp = Comparator.comparing(o -> (char) 0);", - " Comparator cCmp2 = Comparator.comparing(o -> (char) 0);", - " Comparator sCmp = Comparator.comparing(o -> (short) 0);", - " Comparator sCmp2 = Comparator.comparing(o -> (short) 0);", - " Comparator iCmp = Comparator.comparing(o -> 0);", - " Comparator iCmp2 = Comparator.comparing(o -> 0);", - " Comparator lCmp = Comparator.comparing(o -> 0L);", - " Comparator lCmp2 = Comparator.comparing(o -> 0L);", - " Comparator fCmp = Comparator.comparing(o -> 0.0f);", - " Comparator fCmp2 = Comparator.comparing(o -> 0.0f);", - " Comparator dCmp = Comparator.comparing(o -> 0.0);", - " Comparator dCmp2 = Comparator.comparing(o -> 0.0);", - "", - " default void m() {", - " bCmp.thenComparing(o -> (byte) 0);", - " bCmp.thenComparing(o -> (byte) 0);", - " cCmp.thenComparing(o -> (char) 0);", - " cCmp.thenComparing(o -> (char) 0);", - " sCmp.thenComparing(o -> (short) 0);", - " sCmp.thenComparing(o -> (short) 0);", - " iCmp.thenComparing(o -> 0);", - " iCmp.thenComparing(o -> 0);", - " lCmp.thenComparing(o -> 0L);", - " lCmp.thenComparing(o -> 0L);", - " fCmp.thenComparing(o -> 0.0f);", - " fCmp.thenComparing(o -> 0.0f);", - " dCmp.thenComparing(o -> 0.0);", - " dCmp.thenComparing(o -> 0.0);", - " }", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparing(o -> (byte) 0); + Comparator bCmp2 = Comparator.comparing(o -> (byte) 0); + Comparator cCmp = Comparator.comparing(o -> (char) 0); + Comparator cCmp2 = Comparator.comparing(o -> (char) 0); + Comparator sCmp = Comparator.comparing(o -> (short) 0); + Comparator sCmp2 = Comparator.comparing(o -> (short) 0); + Comparator iCmp = Comparator.comparing(o -> 0); + Comparator iCmp2 = Comparator.comparing(o -> 0); + Comparator lCmp = Comparator.comparing(o -> 0L); + Comparator lCmp2 = Comparator.comparing(o -> 0L); + Comparator fCmp = Comparator.comparing(o -> 0.0f); + Comparator fCmp2 = Comparator.comparing(o -> 0.0f); + Comparator dCmp = Comparator.comparing(o -> 0.0); + Comparator dCmp2 = Comparator.comparing(o -> 0.0); + + default void m() { + bCmp.thenComparing(o -> (byte) 0); + bCmp.thenComparing(o -> (byte) 0); + cCmp.thenComparing(o -> (char) 0); + cCmp.thenComparing(o -> (char) 0); + sCmp.thenComparing(o -> (short) 0); + sCmp.thenComparing(o -> (short) 0); + iCmp.thenComparing(o -> 0); + iCmp.thenComparing(o -> 0); + lCmp.thenComparing(o -> 0L); + lCmp.thenComparing(o -> 0L); + fCmp.thenComparing(o -> 0.0f); + fCmp.thenComparing(o -> 0.0f); + dCmp.thenComparing(o -> 0.0); + dCmp.thenComparing(o -> 0.0); + } + } + + """) .addOutputLines( "out/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparingInt(o -> (byte) 0);", - " Comparator bCmp2 = Comparator.comparingInt(o -> (byte) 0);", - " Comparator cCmp = Comparator.comparingInt(o -> (char) 0);", - " Comparator cCmp2 = Comparator.comparingInt(o -> (char) 0);", - " Comparator sCmp = Comparator.comparingInt(o -> (short) 0);", - " Comparator sCmp2 = Comparator.comparingInt(o -> (short) 0);", - " Comparator iCmp = Comparator.comparingInt(o -> 0);", - " Comparator iCmp2 = Comparator.comparingInt(o -> 0);", - " Comparator lCmp = Comparator.comparingLong(o -> 0L);", - " Comparator lCmp2 = Comparator.comparingLong(o -> 0L);", - " Comparator fCmp = Comparator.comparingDouble(o -> 0.0f);", - " Comparator fCmp2 = Comparator.comparingDouble(o -> 0.0f);", - " Comparator dCmp = Comparator.comparingDouble(o -> 0.0);", - " Comparator dCmp2 = Comparator.comparingDouble(o -> 0.0);", - "", - " default void m() {", - " bCmp.thenComparingInt(o -> (byte) 0);", - " bCmp.thenComparingInt(o -> (byte) 0);", - " cCmp.thenComparingInt(o -> (char) 0);", - " cCmp.thenComparingInt(o -> (char) 0);", - " sCmp.thenComparingInt(o -> (short) 0);", - " sCmp.thenComparingInt(o -> (short) 0);", - " iCmp.thenComparingInt(o -> 0);", - " iCmp.thenComparingInt(o -> 0);", - " lCmp.thenComparingLong(o -> 0L);", - " lCmp.thenComparingLong(o -> 0L);", - " fCmp.thenComparingDouble(o -> 0.0f);", - " fCmp.thenComparingDouble(o -> 0.0f);", - " dCmp.thenComparingDouble(o -> 0.0);", - " dCmp.thenComparingDouble(o -> 0.0);", - " }", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparingInt(o -> (byte) 0); + Comparator bCmp2 = Comparator.comparingInt(o -> (byte) 0); + Comparator cCmp = Comparator.comparingInt(o -> (char) 0); + Comparator cCmp2 = Comparator.comparingInt(o -> (char) 0); + Comparator sCmp = Comparator.comparingInt(o -> (short) 0); + Comparator sCmp2 = Comparator.comparingInt(o -> (short) 0); + Comparator iCmp = Comparator.comparingInt(o -> 0); + Comparator iCmp2 = Comparator.comparingInt(o -> 0); + Comparator lCmp = Comparator.comparingLong(o -> 0L); + Comparator lCmp2 = Comparator.comparingLong(o -> 0L); + Comparator fCmp = Comparator.comparingDouble(o -> 0.0f); + Comparator fCmp2 = Comparator.comparingDouble(o -> 0.0f); + Comparator dCmp = Comparator.comparingDouble(o -> 0.0); + Comparator dCmp2 = Comparator.comparingDouble(o -> 0.0); + + default void m() { + bCmp.thenComparingInt(o -> (byte) 0); + bCmp.thenComparingInt(o -> (byte) 0); + cCmp.thenComparingInt(o -> (char) 0); + cCmp.thenComparingInt(o -> (char) 0); + sCmp.thenComparingInt(o -> (short) 0); + sCmp.thenComparingInt(o -> (short) 0); + iCmp.thenComparingInt(o -> 0); + iCmp.thenComparingInt(o -> 0); + lCmp.thenComparingLong(o -> 0L); + lCmp.thenComparingLong(o -> 0L); + fCmp.thenComparingDouble(o -> 0.0f); + fCmp.thenComparingDouble(o -> 0.0f); + dCmp.thenComparingDouble(o -> 0.0); + dCmp.thenComparingDouble(o -> 0.0); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -580,64 +604,70 @@ void replacementWithBoxedVariants() { refactoringTestHelper .addInputLines( "in/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparingInt(o -> Byte.valueOf((byte) 0));", - " Comparator bCmp2 = Comparator.comparingInt(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp = Comparator.comparingInt(o -> Character.valueOf((char) 0));", - " Comparator cCmp2 = Comparator.comparingInt(o -> Character.valueOf((char) 0));", - " Comparator sCmp = Comparator.comparingInt(o -> Short.valueOf((short) 0));", - " Comparator sCmp2 = Comparator.comparingInt(o -> Short.valueOf((short) 0));", - " Comparator iCmp = Comparator.comparingInt(o -> Integer.valueOf(0));", - " Comparator iCmp2 = Comparator.comparingInt(o -> Integer.valueOf(0));", - " Comparator lCmp = Comparator.comparingLong(o -> Long.valueOf(0));", - " Comparator lCmp2 = Comparator.comparingLong(o -> Long.valueOf(0));", - " Comparator fCmp = Comparator.comparingDouble(o -> Float.valueOf(0));", - " Comparator fCmp2 = Comparator.comparingDouble(o -> Float.valueOf(0));", - " Comparator dCmp = Comparator.comparingDouble(o -> Double.valueOf(0));", - " Comparator dCmp2 = Comparator.comparingDouble(o -> Double.valueOf(0));", - "", - " default void m() {", - " bCmp.thenComparingInt(o -> Byte.valueOf((byte) 0));", - " cCmp.thenComparingInt(o -> Character.valueOf((char) 0));", - " sCmp.thenComparingInt(o -> Short.valueOf((short) 0));", - " iCmp.thenComparingInt(o -> Integer.valueOf(0));", - " lCmp.thenComparingLong(o -> Long.valueOf(0));", - " fCmp.thenComparingDouble(o -> Float.valueOf(0));", - " dCmp.thenComparingDouble(o -> Double.valueOf(0));", - " }", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparingInt(o -> Byte.valueOf((byte) 0)); + Comparator bCmp2 = Comparator.comparingInt(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = Comparator.comparingInt(o -> Character.valueOf((char) 0)); + Comparator cCmp2 = Comparator.comparingInt(o -> Character.valueOf((char) 0)); + Comparator sCmp = Comparator.comparingInt(o -> Short.valueOf((short) 0)); + Comparator sCmp2 = Comparator.comparingInt(o -> Short.valueOf((short) 0)); + Comparator iCmp = Comparator.comparingInt(o -> Integer.valueOf(0)); + Comparator iCmp2 = Comparator.comparingInt(o -> Integer.valueOf(0)); + Comparator lCmp = Comparator.comparingLong(o -> Long.valueOf(0)); + Comparator lCmp2 = Comparator.comparingLong(o -> Long.valueOf(0)); + Comparator fCmp = Comparator.comparingDouble(o -> Float.valueOf(0)); + Comparator fCmp2 = Comparator.comparingDouble(o -> Float.valueOf(0)); + Comparator dCmp = Comparator.comparingDouble(o -> Double.valueOf(0)); + Comparator dCmp2 = Comparator.comparingDouble(o -> Double.valueOf(0)); + + default void m() { + bCmp.thenComparingInt(o -> Byte.valueOf((byte) 0)); + cCmp.thenComparingInt(o -> Character.valueOf((char) 0)); + sCmp.thenComparingInt(o -> Short.valueOf((short) 0)); + iCmp.thenComparingInt(o -> Integer.valueOf(0)); + lCmp.thenComparingLong(o -> Long.valueOf(0)); + fCmp.thenComparingDouble(o -> Float.valueOf(0)); + dCmp.thenComparingDouble(o -> Double.valueOf(0)); + } + } + + """) .addOutputLines( "out/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparing(o -> Byte.valueOf((byte) 0));", - " Comparator bCmp2 = Comparator.comparing(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp = Comparator.comparing(o -> Character.valueOf((char) 0));", - " Comparator cCmp2 = Comparator.comparing(o -> Character.valueOf((char) 0));", - " Comparator sCmp = Comparator.comparing(o -> Short.valueOf((short) 0));", - " Comparator sCmp2 = Comparator.comparing(o -> Short.valueOf((short) 0));", - " Comparator iCmp = Comparator.comparing(o -> Integer.valueOf(0));", - " Comparator iCmp2 = Comparator.comparing(o -> Integer.valueOf(0));", - " Comparator lCmp = Comparator.comparing(o -> Long.valueOf(0));", - " Comparator lCmp2 = Comparator.comparing(o -> Long.valueOf(0));", - " Comparator fCmp = Comparator.comparing(o -> Float.valueOf(0));", - " Comparator fCmp2 = Comparator.comparing(o -> Float.valueOf(0));", - " Comparator dCmp = Comparator.comparing(o -> Double.valueOf(0));", - " Comparator dCmp2 = Comparator.comparing(o -> Double.valueOf(0));", - "", - " default void m() {", - " bCmp.thenComparing(o -> Byte.valueOf((byte) 0));", - " cCmp.thenComparing(o -> Character.valueOf((char) 0));", - " sCmp.thenComparing(o -> Short.valueOf((short) 0));", - " iCmp.thenComparing(o -> Integer.valueOf(0));", - " lCmp.thenComparing(o -> Long.valueOf(0));", - " fCmp.thenComparing(o -> Float.valueOf(0));", - " dCmp.thenComparing(o -> Double.valueOf(0));", - " }", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparing(o -> Byte.valueOf((byte) 0)); + Comparator bCmp2 = Comparator.comparing(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = Comparator.comparing(o -> Character.valueOf((char) 0)); + Comparator cCmp2 = Comparator.comparing(o -> Character.valueOf((char) 0)); + Comparator sCmp = Comparator.comparing(o -> Short.valueOf((short) 0)); + Comparator sCmp2 = Comparator.comparing(o -> Short.valueOf((short) 0)); + Comparator iCmp = Comparator.comparing(o -> Integer.valueOf(0)); + Comparator iCmp2 = Comparator.comparing(o -> Integer.valueOf(0)); + Comparator lCmp = Comparator.comparing(o -> Long.valueOf(0)); + Comparator lCmp2 = Comparator.comparing(o -> Long.valueOf(0)); + Comparator fCmp = Comparator.comparing(o -> Float.valueOf(0)); + Comparator fCmp2 = Comparator.comparing(o -> Float.valueOf(0)); + Comparator dCmp = Comparator.comparing(o -> Double.valueOf(0)); + Comparator dCmp2 = Comparator.comparing(o -> Double.valueOf(0)); + + default void m() { + bCmp.thenComparing(o -> Byte.valueOf((byte) 0)); + cCmp.thenComparing(o -> Character.valueOf((char) 0)); + sCmp.thenComparing(o -> Short.valueOf((short) 0)); + iCmp.thenComparing(o -> Integer.valueOf(0)); + lCmp.thenComparing(o -> Long.valueOf(0)); + fCmp.thenComparing(o -> Float.valueOf(0)); + dCmp.thenComparing(o -> Double.valueOf(0)); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -646,37 +676,43 @@ void replacementWithPrimitiveVariantsUsingStaticImports() { refactoringTestHelper .addInputLines( "in/A.java", - "import static java.util.Comparator.comparing;", - "", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = comparing(o -> (byte) 0);", - " Comparator cCmp = comparing(o -> (char) 0);", - " Comparator sCmp = comparing(o -> (short) 0);", - " Comparator iCmp = comparing(o -> 0);", - " Comparator lCmp = comparing(o -> 0L);", - " Comparator fCmp = comparing(o -> 0.0f);", - " Comparator dCmp = comparing(o -> 0.0);", - "}") + """ + import static java.util.Comparator.comparing; + + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = comparing(o -> (byte) 0); + Comparator cCmp = comparing(o -> (char) 0); + Comparator sCmp = comparing(o -> (short) 0); + Comparator iCmp = comparing(o -> 0); + Comparator lCmp = comparing(o -> 0L); + Comparator fCmp = comparing(o -> 0.0f); + Comparator dCmp = comparing(o -> 0.0); + } + + """) .addOutputLines( "out/A.java", - "import static java.util.Comparator.comparing;", - "import static java.util.Comparator.comparingDouble;", - "import static java.util.Comparator.comparingInt;", - "import static java.util.Comparator.comparingLong;", - "", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = comparingInt(o -> (byte) 0);", - " Comparator cCmp = comparingInt(o -> (char) 0);", - " Comparator sCmp = comparingInt(o -> (short) 0);", - " Comparator iCmp = comparingInt(o -> 0);", - " Comparator lCmp = comparingLong(o -> 0L);", - " Comparator fCmp = comparingDouble(o -> 0.0f);", - " Comparator dCmp = comparingDouble(o -> 0.0);", - "}") + """ + import static java.util.Comparator.comparing; + import static java.util.Comparator.comparingDouble; + import static java.util.Comparator.comparingInt; + import static java.util.Comparator.comparingLong; + + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = comparingInt(o -> (byte) 0); + Comparator cCmp = comparingInt(o -> (char) 0); + Comparator sCmp = comparingInt(o -> (short) 0); + Comparator iCmp = comparingInt(o -> 0); + Comparator lCmp = comparingLong(o -> 0L); + Comparator fCmp = comparingDouble(o -> 0.0f); + Comparator dCmp = comparingDouble(o -> 0.0); + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -685,39 +721,45 @@ void replacementWithBoxedVariantsUsingStaticImports() { refactoringTestHelper .addInputLines( "in/A.java", - "import static java.util.Comparator.comparingDouble;", - "import static java.util.Comparator.comparingInt;", - "import static java.util.Comparator.comparingLong;", - "", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = comparingInt(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp = comparingInt(o -> Character.valueOf((char) 0));", - " Comparator sCmp = comparingInt(o -> Short.valueOf((short) 0));", - " Comparator iCmp = comparingInt(o -> Integer.valueOf(0));", - " Comparator lCmp = comparingLong(o -> Long.valueOf(0));", - " Comparator fCmp = comparingDouble(o -> Float.valueOf(0));", - " Comparator dCmp = comparingDouble(o -> Double.valueOf(0));", - "}") + """ + import static java.util.Comparator.comparingDouble; + import static java.util.Comparator.comparingInt; + import static java.util.Comparator.comparingLong; + + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = comparingInt(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = comparingInt(o -> Character.valueOf((char) 0)); + Comparator sCmp = comparingInt(o -> Short.valueOf((short) 0)); + Comparator iCmp = comparingInt(o -> Integer.valueOf(0)); + Comparator lCmp = comparingLong(o -> Long.valueOf(0)); + Comparator fCmp = comparingDouble(o -> Float.valueOf(0)); + Comparator dCmp = comparingDouble(o -> Double.valueOf(0)); + } + + """) .addOutputLines( "out/A.java", - "import static java.util.Comparator.comparing;", - "import static java.util.Comparator.comparingDouble;", - "import static java.util.Comparator.comparingInt;", - "import static java.util.Comparator.comparingLong;", - "", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = comparing(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp = comparing(o -> Character.valueOf((char) 0));", - " Comparator sCmp = comparing(o -> Short.valueOf((short) 0));", - " Comparator iCmp = comparing(o -> Integer.valueOf(0));", - " Comparator lCmp = comparing(o -> Long.valueOf(0));", - " Comparator fCmp = comparing(o -> Float.valueOf(0));", - " Comparator dCmp = comparing(o -> Double.valueOf(0));", - "}") + """ + import static java.util.Comparator.comparing; + import static java.util.Comparator.comparingDouble; + import static java.util.Comparator.comparingInt; + import static java.util.Comparator.comparingLong; + + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = comparing(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = comparing(o -> Character.valueOf((char) 0)); + Comparator sCmp = comparing(o -> Short.valueOf((short) 0)); + Comparator iCmp = comparing(o -> Integer.valueOf(0)); + Comparator lCmp = comparing(o -> Long.valueOf(0)); + Comparator fCmp = comparing(o -> Float.valueOf(0)); + Comparator dCmp = comparing(o -> Double.valueOf(0)); + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -726,30 +768,36 @@ void replacementWithPrimitiveVariantsInComplexSyntacticalContext() { refactoringTestHelper .addInputLines( "in/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparing(o -> o).thenComparing(o -> (byte) 0);", - " Comparator cCmp = Comparator.comparing(o -> o).thenComparing(o -> (char) 0);", - " Comparator sCmp = Comparator.comparing(o -> o).thenComparing(o -> (short) 0);", - " Comparator iCmp = Comparator.comparing(o -> o).thenComparing(o -> 0);", - " Comparator lCmp = Comparator.comparing(o -> o).thenComparing(o -> 0L);", - " Comparator fCmp = Comparator.comparing(o -> o).thenComparing(o -> 0.0f);", - " Comparator dCmp = Comparator.comparing(o -> o).thenComparing(o -> 0.0);", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparing(o -> o).thenComparing(o -> (byte) 0); + Comparator cCmp = Comparator.comparing(o -> o).thenComparing(o -> (char) 0); + Comparator sCmp = Comparator.comparing(o -> o).thenComparing(o -> (short) 0); + Comparator iCmp = Comparator.comparing(o -> o).thenComparing(o -> 0); + Comparator lCmp = Comparator.comparing(o -> o).thenComparing(o -> 0L); + Comparator fCmp = Comparator.comparing(o -> o).thenComparing(o -> 0.0f); + Comparator dCmp = Comparator.comparing(o -> o).thenComparing(o -> 0.0); + } + + """) .addOutputLines( "out/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (byte) 0);", - " Comparator cCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (char) 0);", - " Comparator sCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (short) 0);", - " Comparator iCmp = Comparator.comparing(o -> o).thenComparingInt(o -> 0);", - " Comparator lCmp = Comparator.comparing(o -> o).thenComparingLong(o -> 0L);", - " Comparator fCmp = Comparator.comparing(o -> o).thenComparingDouble(o -> 0.0f);", - " Comparator dCmp = Comparator.comparing(o -> o).thenComparingDouble(o -> 0.0);", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (byte) 0); + Comparator cCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (char) 0); + Comparator sCmp = Comparator.comparing(o -> o).thenComparingInt(o -> (short) 0); + Comparator iCmp = Comparator.comparing(o -> o).thenComparingInt(o -> 0); + Comparator lCmp = Comparator.comparing(o -> o).thenComparingLong(o -> 0L); + Comparator fCmp = Comparator.comparing(o -> o).thenComparingDouble(o -> 0.0f); + Comparator dCmp = Comparator.comparing(o -> o).thenComparingDouble(o -> 0.0); + } + + """) .doTest(TestMode.TEXT_MATCH); } @@ -758,38 +806,44 @@ void replacementWithBoxedVariantsInComplexSyntacticalContext() { refactoringTestHelper .addInputLines( "in/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp =", - " Comparator.comparing(o -> o).thenComparingInt(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp =", - " Comparator.comparing(o -> o).thenComparingInt(o -> Character.valueOf((char) 0));", - " Comparator sCmp =", - " Comparator.comparing(o -> o).thenComparingInt(o -> Short.valueOf((short) 0));", - " Comparator iCmp = Comparator.comparing(o -> o).thenComparingInt(o -> Integer.valueOf(0));", - " Comparator lCmp = Comparator.comparing(o -> o).thenComparingLong(o -> Long.valueOf(0));", - " Comparator fCmp =", - " Comparator.comparing(o -> o).thenComparingDouble(o -> Float.valueOf(0));", - " Comparator dCmp =", - " Comparator.comparing(o -> o).thenComparingDouble(o -> Double.valueOf(0));", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = + Comparator.comparing(o -> o).thenComparingInt(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = + Comparator.comparing(o -> o).thenComparingInt(o -> Character.valueOf((char) 0)); + Comparator sCmp = + Comparator.comparing(o -> o).thenComparingInt(o -> Short.valueOf((short) 0)); + Comparator iCmp = Comparator.comparing(o -> o).thenComparingInt(o -> Integer.valueOf(0)); + Comparator lCmp = Comparator.comparing(o -> o).thenComparingLong(o -> Long.valueOf(0)); + Comparator fCmp = + Comparator.comparing(o -> o).thenComparingDouble(o -> Float.valueOf(0)); + Comparator dCmp = + Comparator.comparing(o -> o).thenComparingDouble(o -> Double.valueOf(0)); + } + + """) .addOutputLines( "out/A.java", - "import java.util.Comparator;", - "", - "interface A extends Comparable {", - " Comparator bCmp =", - " Comparator.comparing(o -> o).thenComparing(o -> Byte.valueOf((byte) 0));", - " Comparator cCmp =", - " Comparator.comparing(o -> o).thenComparing(o -> Character.valueOf((char) 0));", - " Comparator sCmp =", - " Comparator.comparing(o -> o).thenComparing(o -> Short.valueOf((short) 0));", - " Comparator iCmp = Comparator.comparing(o -> o).thenComparing(o -> Integer.valueOf(0));", - " Comparator lCmp = Comparator.comparing(o -> o).thenComparing(o -> Long.valueOf(0));", - " Comparator fCmp = Comparator.comparing(o -> o).thenComparing(o -> Float.valueOf(0));", - " Comparator dCmp = Comparator.comparing(o -> o).thenComparing(o -> Double.valueOf(0));", - "}") + """ + import java.util.Comparator; + + interface A extends Comparable { + Comparator bCmp = + Comparator.comparing(o -> o).thenComparing(o -> Byte.valueOf((byte) 0)); + Comparator cCmp = + Comparator.comparing(o -> o).thenComparing(o -> Character.valueOf((char) 0)); + Comparator sCmp = + Comparator.comparing(o -> o).thenComparing(o -> Short.valueOf((short) 0)); + Comparator iCmp = Comparator.comparing(o -> o).thenComparing(o -> Integer.valueOf(0)); + Comparator lCmp = Comparator.comparing(o -> o).thenComparing(o -> Long.valueOf(0)); + Comparator fCmp = Comparator.comparing(o -> o).thenComparing(o -> Float.valueOf(0)); + Comparator dCmp = Comparator.comparing(o -> o).thenComparing(o -> Double.valueOf(0)); + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RedundantStringConversionTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RedundantStringConversionTest.java index 18c0ebea7f2..9d780b0503a 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RedundantStringConversionTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RedundantStringConversionTest.java @@ -25,21 +25,24 @@ void identificationOfIdentityTransformation() { compilationTestHelper .addSourceLines( "A.java", - "class A {", - " private final Object o = new Object();", - " private final String s = o.toString();", - "", - " String[] m() {", - " return new String[] {", - " o.toString(),", - " // BUG: Diagnostic contains:", - " s.toString(),", - " String.valueOf(o),", - " // BUG: Diagnostic contains:", - " String.valueOf(s),", - " };", - " }", - "}") + """ + class A { + private final Object o = new Object(); + private final String s = o.toString(); + + String[] m() { + return new String[] { + o.toString(), + // BUG: Diagnostic contains: + s.toString(), + String.valueOf(o), + // BUG: Diagnostic contains: + String.valueOf(s), + }; + } + } + + """) .doTest(); } @@ -48,54 +51,57 @@ void identificationWithinMutatingAssignment() { compilationTestHelper .addSourceLines( "A.java", - "import java.math.BigInteger;", - "import java.util.Objects;", - "", - "class A {", - " private final BigInteger i = BigInteger.ZERO;", - "", - " void m1() {", - " String s = i.toString();", - " // BUG: Diagnostic contains:", - " s += this.toString();", - " s += super.toString();", - " // BUG: Diagnostic contains:", - " s += i.toString();", - " s += i.toString(16);", - " // BUG: Diagnostic contains:", - " s += Objects.toString(i);", - " // BUG: Diagnostic contains:", - " s += Objects.toString(null);", - " // BUG: Diagnostic contains:", - " s += String.valueOf(i);", - " // BUG: Diagnostic contains:", - " s += String.valueOf((String) null);", - " s += String.valueOf(null);", - " s += String.valueOf(new char[0]);", - " // BUG: Diagnostic contains:", - " s += Byte.toString((byte) 0);", - " // BUG: Diagnostic contains:", - " s += Character.toString((char) 0);", - " // BUG: Diagnostic contains:", - " s += Short.toString((short) 0);", - " // BUG: Diagnostic contains:", - " s += Integer.toString(0);", - " // BUG: Diagnostic contains:", - " s += Long.toString(0);", - " // BUG: Diagnostic contains:", - " s += Float.toString((float) 0.0);", - " // BUG: Diagnostic contains:", - " s += Double.toString(0.0);", - " }", - "", - " void m2() {", - " int i = 0;", - " i += 1;", - " i -= 1;", - " i *= 1;", - " i /= 1;", - " }", - "}") + """ + import java.math.BigInteger; + import java.util.Objects; + + class A { + private final BigInteger i = BigInteger.ZERO; + + void m1() { + String s = i.toString(); + // BUG: Diagnostic contains: + s += this.toString(); + s += super.toString(); + // BUG: Diagnostic contains: + s += i.toString(); + s += i.toString(16); + // BUG: Diagnostic contains: + s += Objects.toString(i); + // BUG: Diagnostic contains: + s += Objects.toString(null); + // BUG: Diagnostic contains: + s += String.valueOf(i); + // BUG: Diagnostic contains: + s += String.valueOf((String) null); + s += String.valueOf(null); + s += String.valueOf(new char[0]); + // BUG: Diagnostic contains: + s += Byte.toString((byte) 0); + // BUG: Diagnostic contains: + s += Character.toString((char) 0); + // BUG: Diagnostic contains: + s += Short.toString((short) 0); + // BUG: Diagnostic contains: + s += Integer.toString(0); + // BUG: Diagnostic contains: + s += Long.toString(0); + // BUG: Diagnostic contains: + s += Float.toString((float) 0.0); + // BUG: Diagnostic contains: + s += Double.toString(0.0); + } + + void m2() { + int i = 0; + i += 1; + i -= 1; + i *= 1; + i /= 1; + } + } + + """) .doTest(); } @@ -104,81 +110,84 @@ void identificationWithinBinaryOperation() { compilationTestHelper .addSourceLines( "A.java", - "import java.math.BigInteger;", - "", - "class A {", - " private final BigInteger i = BigInteger.ZERO;", - " private final String s = i.toString();", - "", - " String[] m1() {", - " return new String[] {", - " // BUG: Diagnostic contains:", - " s + this.toString(),", - " s + super.toString(),", - " // BUG: Diagnostic contains:", - " s + i.toString(),", - " s + i.toString(16),", - " // BUG: Diagnostic contains:", - " s + String.valueOf(i),", - " // BUG: Diagnostic contains:", - " s + String.valueOf((String) null),", - " s + String.valueOf(null),", - " s + String.valueOf(new char[0]),", - " //", - " 42 + this.toString(),", - " 42 + super.toString(),", - " 42 + i.toString(),", - " 42 + i.toString(16),", - " 42 + String.valueOf(i),", - " // BUG: Diagnostic contains:", - " 42 + String.valueOf((String) null),", - " 42 + String.valueOf(null),", - " 42 + String.valueOf(new char[0]),", - "", - " // BUG: Diagnostic contains:", - " this.toString() + s,", - " super.toString() + s,", - " // BUG: Diagnostic contains:", - " i.toString() + s,", - " i.toString(16) + s,", - " // BUG: Diagnostic contains:", - " String.valueOf(i) + s,", - " // BUG: Diagnostic contains:", - " String.valueOf((String) null) + s,", - " String.valueOf(null) + s,", - " String.valueOf(new char[0]) + s,", - " //", - " this.toString() + 42,", - " super.toString() + 42,", - " i.toString() + 42,", - " i.toString(16) + 42,", - " String.valueOf(i) + 42,", - " // BUG: Diagnostic contains:", - " String.valueOf((String) null) + 42,", - " String.valueOf(null) + 42,", - " String.valueOf(new char[0]) + 42,", - "", - " // BUG: Diagnostic contains:", - " this.toString() + this.toString(),", - " super.toString() + super.toString(),", - " // BUG: Diagnostic contains:", - " i.toString() + i.toString(),", - " i.toString(16) + i.toString(16),", - " // BUG: Diagnostic contains:", - " String.valueOf(i) + String.valueOf(i),", - " // BUG: Diagnostic contains:", - " String.valueOf((String) null) + String.valueOf((String) null),", - " String.valueOf(null) + String.valueOf(null),", - " String.valueOf(new char[0]) + String.valueOf(new char[0]),", - " };", - " }", - "", - " int[] m2() {", - " return new int[] {", - " 1 + 1, 1 - 1, 1 * 1, 1 / 1,", - " };", - " }", - "}") + """ + import java.math.BigInteger; + + class A { + private final BigInteger i = BigInteger.ZERO; + private final String s = i.toString(); + + String[] m1() { + return new String[] { + // BUG: Diagnostic contains: + s + this.toString(), + s + super.toString(), + // BUG: Diagnostic contains: + s + i.toString(), + s + i.toString(16), + // BUG: Diagnostic contains: + s + String.valueOf(i), + // BUG: Diagnostic contains: + s + String.valueOf((String) null), + s + String.valueOf(null), + s + String.valueOf(new char[0]), + // + 42 + this.toString(), + 42 + super.toString(), + 42 + i.toString(), + 42 + i.toString(16), + 42 + String.valueOf(i), + // BUG: Diagnostic contains: + 42 + String.valueOf((String) null), + 42 + String.valueOf(null), + 42 + String.valueOf(new char[0]), + + // BUG: Diagnostic contains: + this.toString() + s, + super.toString() + s, + // BUG: Diagnostic contains: + i.toString() + s, + i.toString(16) + s, + // BUG: Diagnostic contains: + String.valueOf(i) + s, + // BUG: Diagnostic contains: + String.valueOf((String) null) + s, + String.valueOf(null) + s, + String.valueOf(new char[0]) + s, + // + this.toString() + 42, + super.toString() + 42, + i.toString() + 42, + i.toString(16) + 42, + String.valueOf(i) + 42, + // BUG: Diagnostic contains: + String.valueOf((String) null) + 42, + String.valueOf(null) + 42, + String.valueOf(new char[0]) + 42, + + // BUG: Diagnostic contains: + this.toString() + this.toString(), + super.toString() + super.toString(), + // BUG: Diagnostic contains: + i.toString() + i.toString(), + i.toString(16) + i.toString(16), + // BUG: Diagnostic contains: + String.valueOf(i) + String.valueOf(i), + // BUG: Diagnostic contains: + String.valueOf((String) null) + String.valueOf((String) null), + String.valueOf(null) + String.valueOf(null), + String.valueOf(new char[0]) + String.valueOf(new char[0]), + }; + } + + int[] m2() { + return new int[] { + 1 + 1, 1 - 1, 1 * 1, 1 / 1, + }; + } + } + + """) .doTest(); } @@ -187,46 +196,49 @@ void identificationWithinStringBuilderMethod() { compilationTestHelper .addSourceLines( "A.java", - "import java.math.BigInteger;", - "", - "class A {", - " private final BigInteger i = BigInteger.ZERO;", - " private final String s = i.toString();", - "", - " void m() {", - " StringBuilder sb = new StringBuilder();", - "", - " sb.append(1);", - " sb.append(i);", - " // BUG: Diagnostic contains:", - " sb.append(i.toString());", - " sb.append(i.toString(16));", - " // BUG: Diagnostic contains:", - " sb.append(String.valueOf(i));", - " // BUG: Diagnostic contains:", - " sb.append(String.valueOf((String) null));", - " sb.append(String.valueOf(null));", - " sb.append(String.valueOf(new char[0]));", - " sb.append(s);", - " sb.append(\"constant\");", - "", - " sb.insert(0, 1);", - " sb.insert(0, i);", - " // BUG: Diagnostic contains:", - " sb.insert(0, i.toString());", - " sb.insert(0, i.toString(16));", - " // BUG: Diagnostic contains:", - " sb.insert(0, String.valueOf(i));", - " // BUG: Diagnostic contains:", - " sb.insert(0, String.valueOf((String) null));", - " sb.insert(0, String.valueOf(null));", - " sb.insert(0, String.valueOf(new char[0]));", - " sb.insert(0, s);", - " sb.insert(0, \"constant\");", - "", - " sb.replace(0, 1, i.toString());", - " }", - "}") + """ + import java.math.BigInteger; + + class A { + private final BigInteger i = BigInteger.ZERO; + private final String s = i.toString(); + + void m() { + StringBuilder sb = new StringBuilder(); + + sb.append(1); + sb.append(i); + // BUG: Diagnostic contains: + sb.append(i.toString()); + sb.append(i.toString(16)); + // BUG: Diagnostic contains: + sb.append(String.valueOf(i)); + // BUG: Diagnostic contains: + sb.append(String.valueOf((String) null)); + sb.append(String.valueOf(null)); + sb.append(String.valueOf(new char[0])); + sb.append(s); + sb.append("constant"); + + sb.insert(0, 1); + sb.insert(0, i); + // BUG: Diagnostic contains: + sb.insert(0, i.toString()); + sb.insert(0, i.toString(16)); + // BUG: Diagnostic contains: + sb.insert(0, String.valueOf(i)); + // BUG: Diagnostic contains: + sb.insert(0, String.valueOf((String) null)); + sb.insert(0, String.valueOf(null)); + sb.insert(0, String.valueOf(new char[0])); + sb.insert(0, s); + sb.insert(0, "constant"); + + sb.replace(0, 1, i.toString()); + } + } + + """) .doTest(); } @@ -236,43 +248,46 @@ void identificationWithinFormatterMethod() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Formattable;", - "import java.util.Locale;", - "", - "class A {", - " private final Locale locale = Locale.ROOT;", - " private final Formattable f = (formatter, flags, width, precision) -> {};", - " private final Object o = new Object();", - " private final String s = o.toString();", - "", - " void m() {", - " String.format(s, f);", - " String.format(s, o);", - " String.format(s, s);", - " String.format(s, f.toString());", - " // BUG: Diagnostic contains:", - " String.format(s, o.toString());", - " // BUG: Diagnostic contains:", - " String.format(s, String.valueOf(o));", - "", - " String.format(locale, s, f);", - " String.format(locale, s, o);", - " String.format(locale, s, s);", - " String.format(locale, s, f.toString());", - " // BUG: Diagnostic contains:", - " String.format(locale, s, o.toString());", - " // BUG: Diagnostic contains:", - " String.format(locale, s, String.valueOf(o));", - "", - " String.format(o.toString(), o);", - " // BUG: Diagnostic contains:", - " String.format(s.toString(), o);", - " String.format(locale.toString(), s, o);", - " String.format(locale, o.toString(), o);", - " // BUG: Diagnostic contains:", - " String.format(locale, s.toString(), o);", - " }", - "}") + """ + import java.util.Formattable; + import java.util.Locale; + + class A { + private final Locale locale = Locale.ROOT; + private final Formattable f = (formatter, flags, width, precision) -> {}; + private final Object o = new Object(); + private final String s = o.toString(); + + void m() { + String.format(s, f); + String.format(s, o); + String.format(s, s); + String.format(s, f.toString()); + // BUG: Diagnostic contains: + String.format(s, o.toString()); + // BUG: Diagnostic contains: + String.format(s, String.valueOf(o)); + + String.format(locale, s, f); + String.format(locale, s, o); + String.format(locale, s, s); + String.format(locale, s, f.toString()); + // BUG: Diagnostic contains: + String.format(locale, s, o.toString()); + // BUG: Diagnostic contains: + String.format(locale, s, String.valueOf(o)); + + String.format(o.toString(), o); + // BUG: Diagnostic contains: + String.format(s.toString(), o); + String.format(locale.toString(), s, o); + String.format(locale, o.toString(), o); + // BUG: Diagnostic contains: + String.format(locale, s.toString(), o); + } + } + + """) .doTest(); } @@ -281,58 +296,61 @@ void identificationWithinGuavaGuardMethod() { compilationTestHelper .addSourceLines( "A.java", - "import static com.google.common.base.Preconditions.checkArgument;", - "import static com.google.common.base.Preconditions.checkNotNull;", - "import static com.google.common.base.Preconditions.checkState;", - "import static com.google.common.base.Verify.verify;", - "import static com.google.common.base.Verify.verifyNotNull;", - "", - "import java.util.Formattable;", - "", - "class A {", - " private final Formattable f = (formatter, flags, width, precision) -> {};", - " private final Object o = new Object();", - " private final String s = o.toString();", - "", - " void m() {", - " checkState(true, s, f);", - " // BUG: Diagnostic contains:", - " checkState(true, s, f.toString());", - " checkState(true, f.toString(), f);", - " // BUG: Diagnostic contains:", - " checkState(true, s.toString(), f);", - "", - " checkArgument(true, s, f);", - " // BUG: Diagnostic contains:", - " checkArgument(true, s, f.toString());", - " checkArgument(true, f.toString(), f);", - " // BUG: Diagnostic contains:", - " checkArgument(true, s.toString(), f);", - "", - " checkNotNull(o, s, f);", - " // BUG: Diagnostic contains:", - " checkNotNull(o, s, f.toString());", - " checkNotNull(o, f.toString(), f);", - " // BUG: Diagnostic contains:", - " checkNotNull(o, s.toString(), f);", - " checkNotNull(o.toString(), s, f);", - "", - " verify(true, s, f);", - " // BUG: Diagnostic contains:", - " verify(true, s, f.toString());", - " verify(true, f.toString(), f);", - " // BUG: Diagnostic contains:", - " verify(true, s.toString(), f);", - "", - " verifyNotNull(o, s, f);", - " // BUG: Diagnostic contains:", - " verifyNotNull(o, s, f.toString());", - " verifyNotNull(o, f.toString(), f);", - " // BUG: Diagnostic contains:", - " verifyNotNull(o, s.toString(), f);", - " verifyNotNull(o.toString(), s, f);", - " }", - "}") + """ + import static com.google.common.base.Preconditions.checkArgument; + import static com.google.common.base.Preconditions.checkNotNull; + import static com.google.common.base.Preconditions.checkState; + import static com.google.common.base.Verify.verify; + import static com.google.common.base.Verify.verifyNotNull; + + import java.util.Formattable; + + class A { + private final Formattable f = (formatter, flags, width, precision) -> {}; + private final Object o = new Object(); + private final String s = o.toString(); + + void m() { + checkState(true, s, f); + // BUG: Diagnostic contains: + checkState(true, s, f.toString()); + checkState(true, f.toString(), f); + // BUG: Diagnostic contains: + checkState(true, s.toString(), f); + + checkArgument(true, s, f); + // BUG: Diagnostic contains: + checkArgument(true, s, f.toString()); + checkArgument(true, f.toString(), f); + // BUG: Diagnostic contains: + checkArgument(true, s.toString(), f); + + checkNotNull(o, s, f); + // BUG: Diagnostic contains: + checkNotNull(o, s, f.toString()); + checkNotNull(o, f.toString(), f); + // BUG: Diagnostic contains: + checkNotNull(o, s.toString(), f); + checkNotNull(o.toString(), s, f); + + verify(true, s, f); + // BUG: Diagnostic contains: + verify(true, s, f.toString()); + verify(true, f.toString(), f); + // BUG: Diagnostic contains: + verify(true, s.toString(), f); + + verifyNotNull(o, s, f); + // BUG: Diagnostic contains: + verifyNotNull(o, s, f.toString()); + verifyNotNull(o, f.toString(), f); + // BUG: Diagnostic contains: + verifyNotNull(o, s.toString(), f); + verifyNotNull(o.toString(), s, f); + } + } + + """) .doTest(); } @@ -341,53 +359,56 @@ void identificationWithinSlf4jLoggerMethod() { compilationTestHelper .addSourceLines( "A.java", - "import java.util.Formattable;", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "import org.slf4j.MarkerFactory;", - "", - "class A {", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " private final Marker marker = MarkerFactory.getMarker(A.class.getName());", - " private final Formattable f = (formatter, flags, width, precision) -> {};", - " private final Object o = new Object();", - " private final String s = f.toString();", - " private final Throwable t = new Throwable();", - "", - " void m() {", - " LOG.trace(s, f);", - " // BUG: Diagnostic contains:", - " LOG.debug(s, f.toString());", - " LOG.info(s, t.toString());", - " LOG.warn(s, o, t.toString());", - " // BUG: Diagnostic contains:", - " LOG.error(s, o.toString(), t.toString());", - " // BUG: Diagnostic contains:", - " LOG.trace(s, t.toString(), o);", - "", - " LOG.trace(marker, s, f);", - " // BUG: Diagnostic contains:", - " LOG.debug(marker, s, f.toString());", - " LOG.info(marker, s, t.toString());", - " LOG.warn(marker, s, o, t.toString());", - " // BUG: Diagnostic contains:", - " LOG.error(marker, s, o.toString(), t.toString());", - " // BUG: Diagnostic contains:", - " LOG.trace(marker, s, t.toString(), o);", - "", - " LOG.trace(f.toString(), f);", - " // BUG: Diagnostic contains:", - " LOG.debug(s.toString(), f);", - " LOG.info(t.toString(), f);", - " LOG.warn(marker.toString(), s, f);", - " LOG.error(marker, o.toString(), f);", - " // BUG: Diagnostic contains:", - " LOG.trace(marker, s.toString(), f);", - " LOG.debug(marker, t.toString(), f);", - " }", - "}") + """ + import java.util.Formattable; + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + import org.slf4j.MarkerFactory; + + class A { + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + private final Marker marker = MarkerFactory.getMarker(A.class.getName()); + private final Formattable f = (formatter, flags, width, precision) -> {}; + private final Object o = new Object(); + private final String s = f.toString(); + private final Throwable t = new Throwable(); + + void m() { + LOG.trace(s, f); + // BUG: Diagnostic contains: + LOG.debug(s, f.toString()); + LOG.info(s, t.toString()); + LOG.warn(s, o, t.toString()); + // BUG: Diagnostic contains: + LOG.error(s, o.toString(), t.toString()); + // BUG: Diagnostic contains: + LOG.trace(s, t.toString(), o); + + LOG.trace(marker, s, f); + // BUG: Diagnostic contains: + LOG.debug(marker, s, f.toString()); + LOG.info(marker, s, t.toString()); + LOG.warn(marker, s, o, t.toString()); + // BUG: Diagnostic contains: + LOG.error(marker, s, o.toString(), t.toString()); + // BUG: Diagnostic contains: + LOG.trace(marker, s, t.toString(), o); + + LOG.trace(f.toString(), f); + // BUG: Diagnostic contains: + LOG.debug(s.toString(), f); + LOG.info(t.toString(), f); + LOG.warn(marker.toString(), s, f); + LOG.error(marker, o.toString(), f); + // BUG: Diagnostic contains: + LOG.trace(marker, s.toString(), f); + LOG.debug(marker, t.toString(), f); + } + } + + """) .doTest(); } @@ -396,89 +417,92 @@ void identificationOfCustomConversionMethod() { customizedCompilationTestHelper .addSourceLines( "A.java", - "import java.math.RoundingMode;", - "import java.util.Objects;", - "", - "class A {", - " static class B {", - " String name() {", - " return toString();", - " }", - "", - " static String toString(int i) {", - " return Integer.toString(i);", - " }", - "", - " static String toString(int i, int j) {", - " return Integer.toString(i * j);", - " }", - " }", - "", - " enum E {", - " ELEM;", - "", - " public String toString() {", - " return \"__\" + name() + \"__\";", - " }", - " }", - "", - " private final B b = new B();", - " private final String s = b.toString();", - "", - " String[] builtin() {", - " return new String[] {", - " // BUG: Diagnostic contains:", - " s + b.toString(),", - " // BUG: Diagnostic contains:", - " s + Objects.toString(b),", - " // BUG: Diagnostic contains:", - " s + String.valueOf(b),", - " // BUG: Diagnostic contains:", - " s + Byte.toString((byte) 0),", - " // BUG: Diagnostic contains:", - " s + Character.toString((char) 0),", - " // BUG: Diagnostic contains:", - " s + Short.toString((short) 0),", - " // BUG: Diagnostic contains:", - " s + Integer.toString(0),", - " s + Integer.toString(0, 16),", - " // BUG: Diagnostic contains:", - " s + Long.toString(0),", - " s + Long.toString(0, 16),", - " // BUG: Diagnostic contains:", - " s + Float.toString((float) 0.0),", - " // BUG: Diagnostic contains:", - " s + Double.toString(0.0),", - " };", - " }", - "", - " String[] custom() {", - " return new String[] {", - " s + b.name(),", - " // BUG: Diagnostic contains:", - " s + RoundingMode.UP.name(),", - " // BUG: Diagnostic contains:", - " s + mode().name(),", - " s + A.name(),", - " s + A.toString(42),", - " // BUG: Diagnostic contains:", - " s + B.toString(42),", - " s + B.toString(42, 42),", - " };", - " }", - "", - " static String name() {", - " return A.class.toString();", - " }", - "", - " RoundingMode mode() {", - " return RoundingMode.UP;", - " }", - "", - " static String toString(int i) {", - " return Integer.toString(i);", - " }", - "}") + """ + import java.math.RoundingMode; + import java.util.Objects; + + class A { + static class B { + String name() { + return toString(); + } + + static String toString(int i) { + return Integer.toString(i); + } + + static String toString(int i, int j) { + return Integer.toString(i * j); + } + } + + enum E { + ELEM; + + public String toString() { + return "__" + name() + "__"; + } + } + + private final B b = new B(); + private final String s = b.toString(); + + String[] builtin() { + return new String[] { + // BUG: Diagnostic contains: + s + b.toString(), + // BUG: Diagnostic contains: + s + Objects.toString(b), + // BUG: Diagnostic contains: + s + String.valueOf(b), + // BUG: Diagnostic contains: + s + Byte.toString((byte) 0), + // BUG: Diagnostic contains: + s + Character.toString((char) 0), + // BUG: Diagnostic contains: + s + Short.toString((short) 0), + // BUG: Diagnostic contains: + s + Integer.toString(0), + s + Integer.toString(0, 16), + // BUG: Diagnostic contains: + s + Long.toString(0), + s + Long.toString(0, 16), + // BUG: Diagnostic contains: + s + Float.toString((float) 0.0), + // BUG: Diagnostic contains: + s + Double.toString(0.0), + }; + } + + String[] custom() { + return new String[] { + s + b.name(), + // BUG: Diagnostic contains: + s + RoundingMode.UP.name(), + // BUG: Diagnostic contains: + s + mode().name(), + s + A.name(), + s + A.toString(42), + // BUG: Diagnostic contains: + s + B.toString(42), + s + B.toString(42, 42), + }; + } + + static String name() { + return A.class.toString(); + } + + RoundingMode mode() { + return RoundingMode.UP; + } + + static String toString(int i) { + return Integer.toString(i); + } + } + + """) .doTest(); } @@ -487,32 +511,38 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "class A {", - " private final Object o = new Object();", - " private final String s = o.toString();", - "", - " void m() {", - " String v1 = s.toString();", - " String v2 = \"foo\".toString();", - " String v3 = v2 + super.toString();", - " String v4 = 42 + String.valueOf((String) null);", - " String.format(\"%s\", o.toString());", - " }", - "}") + """ + class A { + private final Object o = new Object(); + private final String s = o.toString(); + + void m() { + String v1 = s.toString(); + String v2 = "foo".toString(); + String v3 = v2 + super.toString(); + String v4 = 42 + String.valueOf((String) null); + String.format("%s", o.toString()); + } + } + + """) .addOutputLines( "out/A.java", - "class A {", - " private final Object o = new Object();", - " private final String s = o.toString();", - "", - " void m() {", - " String v1 = s;", - " String v2 = \"foo\";", - " String v3 = v2 + super.toString();", - " String v4 = 42 + (String) null;", - " String.format(\"%s\", o);", - " }", - "}") + """ + class A { + private final Object o = new Object(); + private final String s = o.toString(); + + void m() { + String v1 = s; + String v2 = "foo"; + String v3 = v2 + super.toString(); + String v4 = 42 + (String) null; + String.format("%s", o); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RefasterAnyOfUsageTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RefasterAnyOfUsageTest.java index fd10444a41c..e31221f11de 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RefasterAnyOfUsageTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RefasterAnyOfUsageTest.java @@ -16,23 +16,26 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.google.errorprone.refaster.Refaster;", - "import com.google.errorprone.refaster.annotation.BeforeTemplate;", - "", - "class A {", - " @BeforeTemplate", - " String before(String str) {", - " // BUG: Diagnostic contains:", - " Refaster.anyOf();", - " // BUG: Diagnostic contains:", - " return Refaster.anyOf(str);", - " }", - "", - " @BeforeTemplate", - " Object before2(String str, Object obj) {", - " return Refaster.anyOf(str, obj);", - " }", - "}") + """ + import com.google.errorprone.refaster.Refaster; + import com.google.errorprone.refaster.annotation.BeforeTemplate; + + class A { + @BeforeTemplate + String before(String str) { + // BUG: Diagnostic contains: + Refaster.anyOf(); + // BUG: Diagnostic contains: + return Refaster.anyOf(str); + } + + @BeforeTemplate + Object before2(String str, Object obj) { + return Refaster.anyOf(str, obj); + } + } + + """) .doTest(); } @@ -41,28 +44,34 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import com.google.errorprone.refaster.Refaster;", - "import com.google.errorprone.refaster.annotation.BeforeTemplate;", - "", - "class A {", - " @BeforeTemplate", - " String before(String str) {", - " Refaster.anyOf();", - " return Refaster.anyOf(str);", - " }", - "}") + """ + import com.google.errorprone.refaster.Refaster; + import com.google.errorprone.refaster.annotation.BeforeTemplate; + + class A { + @BeforeTemplate + String before(String str) { + Refaster.anyOf(); + return Refaster.anyOf(str); + } + } + + """) .addOutputLines( "out/A.java", - "import com.google.errorprone.refaster.Refaster;", - "import com.google.errorprone.refaster.annotation.BeforeTemplate;", - "", - "class A {", - " @BeforeTemplate", - " String before(String str) {", - " Refaster.anyOf();", - " return str;", - " }", - "}") + """ + import com.google.errorprone.refaster.Refaster; + import com.google.errorprone.refaster.annotation.BeforeTemplate; + + class A { + @BeforeTemplate + String before(String str) { + Refaster.anyOf(); + return str; + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestMappingAnnotationTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestMappingAnnotationTest.java index bf34c7bee28..413bef562df 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestMappingAnnotationTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestMappingAnnotationTest.java @@ -12,121 +12,124 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import java.io.InputStream;", - "import java.time.ZoneId;", - "import java.util.Locale;", - "import java.util.TimeZone;", - "import javax.servlet.http.HttpServletRequest;", - "import javax.servlet.http.HttpServletResponse;", - "import org.springframework.http.HttpMethod;", - "import org.springframework.web.bind.annotation.DeleteMapping;", - "import org.springframework.web.bind.annotation.GetMapping;", - "import org.springframework.web.bind.annotation.PatchMapping;", - "import org.springframework.web.bind.annotation.PathVariable;", - "import org.springframework.web.bind.annotation.PostMapping;", - "import org.springframework.web.bind.annotation.PutMapping;", - "import org.springframework.web.bind.annotation.RequestAttribute;", - "import org.springframework.web.bind.annotation.RequestBody;", - "import org.springframework.web.bind.annotation.RequestHeader;", - "import org.springframework.web.bind.annotation.RequestMapping;", - "import org.springframework.web.bind.annotation.RequestParam;", - "import org.springframework.web.context.request.NativeWebRequest;", - "import org.springframework.web.context.request.WebRequest;", - "import org.springframework.web.server.ServerWebExchange;", - "import org.springframework.web.util.UriBuilder;", - "import org.springframework.web.util.UriComponentsBuilder;", - "", - "interface A {", - " A noMapping();", - "", - " A noMapping(String param);", - "", - " @DeleteMapping", - " A properNoParameters();", - "", - " @GetMapping", - " A properPathVariable(@PathVariable String param);", - "", - " @PatchMapping", - " A properRequestAttribute(@RequestAttribute String attribute);", - "", - " @PostMapping", - " A properRequestBody(@RequestBody String body);", - "", - " @PutMapping", - " A properRequestHeader(@RequestHeader String header);", - "", - " @RequestMapping", - " A properRequestParam(@RequestParam String param);", - "", - " @RequestMapping", - " A properInputStream(InputStream input);", - "", - " @RequestMapping", - " A properZoneId(ZoneId zoneId);", - "", - " @RequestMapping", - " A properLocale(Locale locale);", - "", - " @RequestMapping", - " A properTimeZone(TimeZone timeZone);", - "", - " @RequestMapping", - " A properHttpServletRequest(HttpServletRequest request);", - "", - " @RequestMapping", - " A properHttpServletResponse(HttpServletResponse response);", - "", - " @RequestMapping", - " A properHttpMethod(HttpMethod method);", - "", - " @RequestMapping", - " A properNativeWebRequest(NativeWebRequest request);", - "", - " @RequestMapping", - " A properWebRequest(WebRequest request);", - "", - " @RequestMapping", - " A properServerWebExchange(ServerWebExchange exchange);", - "", - " @RequestMapping", - " A properServerUriBuilder(UriBuilder builder);", - "", - " @RequestMapping", - " A properServerUriComponentsBuilder(UriComponentsBuilder builder);", - "", - " @DeleteMapping", - " // BUG: Diagnostic contains:", - " A delete(String param);", - "", - " @GetMapping", - " // BUG: Diagnostic contains:", - " A get(String param);", - "", - " @PatchMapping", - " // BUG: Diagnostic contains:", - " A patch(String param);", - "", - " @PostMapping", - " // BUG: Diagnostic contains:", - " A post(String param);", - "", - " @PutMapping", - " // BUG: Diagnostic contains:", - " A put(String param);", - "", - " @RequestMapping", - " // BUG: Diagnostic contains:", - " A requestMultiple(String param, String param2);", - "", - " @RequestMapping", - " // BUG: Diagnostic contains:", - " A requestFirstParamViolation(String param, @PathVariable String param2);", - "", - " @RequestMapping", - " // BUG: Diagnostic contains:", - " A requestSecondParamViolation(@RequestBody String param, String param2);", - "}") + """ + import java.io.InputStream; + import java.time.ZoneId; + import java.util.Locale; + import java.util.TimeZone; + import javax.servlet.http.HttpServletRequest; + import javax.servlet.http.HttpServletResponse; + import org.springframework.http.HttpMethod; + import org.springframework.web.bind.annotation.DeleteMapping; + import org.springframework.web.bind.annotation.GetMapping; + import org.springframework.web.bind.annotation.PatchMapping; + import org.springframework.web.bind.annotation.PathVariable; + import org.springframework.web.bind.annotation.PostMapping; + import org.springframework.web.bind.annotation.PutMapping; + import org.springframework.web.bind.annotation.RequestAttribute; + import org.springframework.web.bind.annotation.RequestBody; + import org.springframework.web.bind.annotation.RequestHeader; + import org.springframework.web.bind.annotation.RequestMapping; + import org.springframework.web.bind.annotation.RequestParam; + import org.springframework.web.context.request.NativeWebRequest; + import org.springframework.web.context.request.WebRequest; + import org.springframework.web.server.ServerWebExchange; + import org.springframework.web.util.UriBuilder; + import org.springframework.web.util.UriComponentsBuilder; + + interface A { + A noMapping(); + + A noMapping(String param); + + @DeleteMapping + A properNoParameters(); + + @GetMapping + A properPathVariable(@PathVariable String param); + + @PatchMapping + A properRequestAttribute(@RequestAttribute String attribute); + + @PostMapping + A properRequestBody(@RequestBody String body); + + @PutMapping + A properRequestHeader(@RequestHeader String header); + + @RequestMapping + A properRequestParam(@RequestParam String param); + + @RequestMapping + A properInputStream(InputStream input); + + @RequestMapping + A properZoneId(ZoneId zoneId); + + @RequestMapping + A properLocale(Locale locale); + + @RequestMapping + A properTimeZone(TimeZone timeZone); + + @RequestMapping + A properHttpServletRequest(HttpServletRequest request); + + @RequestMapping + A properHttpServletResponse(HttpServletResponse response); + + @RequestMapping + A properHttpMethod(HttpMethod method); + + @RequestMapping + A properNativeWebRequest(NativeWebRequest request); + + @RequestMapping + A properWebRequest(WebRequest request); + + @RequestMapping + A properServerWebExchange(ServerWebExchange exchange); + + @RequestMapping + A properServerUriBuilder(UriBuilder builder); + + @RequestMapping + A properServerUriComponentsBuilder(UriComponentsBuilder builder); + + @DeleteMapping + // BUG: Diagnostic contains: + A delete(String param); + + @GetMapping + // BUG: Diagnostic contains: + A get(String param); + + @PatchMapping + // BUG: Diagnostic contains: + A patch(String param); + + @PostMapping + // BUG: Diagnostic contains: + A post(String param); + + @PutMapping + // BUG: Diagnostic contains: + A put(String param); + + @RequestMapping + // BUG: Diagnostic contains: + A requestMultiple(String param, String param2); + + @RequestMapping + // BUG: Diagnostic contains: + A requestFirstParamViolation(String param, @PathVariable String param2); + + @RequestMapping + // BUG: Diagnostic contains: + A requestSecondParamViolation(@RequestBody String param, String param2); + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestParamTypeTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestParamTypeTest.java index 9020a9c98a7..06ceac99794 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestParamTypeTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/RequestParamTypeTest.java @@ -12,55 +12,58 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.google.common.collect.ImmutableBiMap;", - "import com.google.common.collect.ImmutableList;", - "import com.google.common.collect.ImmutableMap;", - "import com.google.common.collect.ImmutableSet;", - "import java.util.List;", - "import java.util.Map;", - "import java.util.Set;", - "import javax.annotation.Nullable;", - "import org.springframework.web.bind.annotation.DeleteMapping;", - "import org.springframework.web.bind.annotation.GetMapping;", - "import org.springframework.web.bind.annotation.PostMapping;", - "import org.springframework.web.bind.annotation.PutMapping;", - "import org.springframework.web.bind.annotation.RequestBody;", - "import org.springframework.web.bind.annotation.RequestParam;", - "", - "interface A {", - " @PostMapping", - " A properRequestParam(@RequestBody String body);", - "", - " @GetMapping", - " A properRequestParam(@RequestParam int param);", - "", - " @GetMapping", - " A properRequestParam(@RequestParam List param);", - "", - " @PostMapping", - " A properRequestParam(@RequestBody String body, @RequestParam Set param);", - "", - " @PutMapping", - " A properRequestParam(@RequestBody String body, @RequestParam Map param);", - "", - " @GetMapping", - " // BUG: Diagnostic contains:", - " A get(@RequestParam ImmutableBiMap param);", - "", - " @PostMapping", - " // BUG: Diagnostic contains:", - " A post(@Nullable @RequestParam ImmutableList param);", - "", - " @PutMapping", - " // BUG: Diagnostic contains:", - " A put(@RequestBody String body, @RequestParam ImmutableSet param);", - "", - " @DeleteMapping", - " // BUG: Diagnostic contains:", - " A delete(@RequestBody String body, @RequestParam ImmutableMap param);", - "", - " void negative(ImmutableSet set, ImmutableMap map);", - "}") + """ + import com.google.common.collect.ImmutableBiMap; + import com.google.common.collect.ImmutableList; + import com.google.common.collect.ImmutableMap; + import com.google.common.collect.ImmutableSet; + import java.util.List; + import java.util.Map; + import java.util.Set; + import javax.annotation.Nullable; + import org.springframework.web.bind.annotation.DeleteMapping; + import org.springframework.web.bind.annotation.GetMapping; + import org.springframework.web.bind.annotation.PostMapping; + import org.springframework.web.bind.annotation.PutMapping; + import org.springframework.web.bind.annotation.RequestBody; + import org.springframework.web.bind.annotation.RequestParam; + + interface A { + @PostMapping + A properRequestParam(@RequestBody String body); + + @GetMapping + A properRequestParam(@RequestParam int param); + + @GetMapping + A properRequestParam(@RequestParam List param); + + @PostMapping + A properRequestParam(@RequestBody String body, @RequestParam Set param); + + @PutMapping + A properRequestParam(@RequestBody String body, @RequestParam Map param); + + @GetMapping + // BUG: Diagnostic contains: + A get(@RequestParam ImmutableBiMap param); + + @PostMapping + // BUG: Diagnostic contains: + A post(@Nullable @RequestParam ImmutableList param); + + @PutMapping + // BUG: Diagnostic contains: + A put(@RequestBody String body, @RequestParam ImmutableSet param); + + @DeleteMapping + // BUG: Diagnostic contains: + A delete(@RequestBody String body, @RequestParam ImmutableMap param); + + void negative(ImmutableSet set, ImmutableMap map); + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ScheduledTransactionTraceTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ScheduledTransactionTraceTest.java index 3b37b2c0b66..ca762af3bf4 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ScheduledTransactionTraceTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/ScheduledTransactionTraceTest.java @@ -16,30 +16,33 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import com.newrelic.api.agent.Trace;", - "import org.springframework.scheduling.annotation.Scheduled;", - "", - "class A {", - " void notScheduled() {}", - "", - " @Scheduled(fixedDelay = 1)", - " // BUG: Diagnostic contains:", - " void scheduledButNotTraced() {}", - "", - " @Scheduled(fixedDelay = 1)", - " // BUG: Diagnostic contains:", - " @Trace", - " void scheduledButImproperlyTraced1() {}", - "", - " @Scheduled(fixedDelay = 1)", - " // BUG: Diagnostic contains:", - " @Trace(dispatcher = false)", - " void scheduledButImproperlyTraced2() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(dispatcher = true)", - " void scheduledAndProperlyTraced() {}", - "}") + """ + import com.newrelic.api.agent.Trace; + import org.springframework.scheduling.annotation.Scheduled; + + class A { + void notScheduled() {} + + @Scheduled(fixedDelay = 1) + // BUG: Diagnostic contains: + void scheduledButNotTraced() {} + + @Scheduled(fixedDelay = 1) + // BUG: Diagnostic contains: + @Trace + void scheduledButImproperlyTraced1() {} + + @Scheduled(fixedDelay = 1) + // BUG: Diagnostic contains: + @Trace(dispatcher = false) + void scheduledButImproperlyTraced2() {} + + @Scheduled(fixedDelay = 1) + @Trace(dispatcher = true) + void scheduledAndProperlyTraced() {} + } + + """) .doTest(); } @@ -48,47 +51,53 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import com.newrelic.api.agent.Trace;", - "import org.springframework.scheduling.annotation.Scheduled;", - "", - "class A {", - " @Scheduled(fixedDelay = 1)", - " void scheduledButNotTraced() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace", - " void scheduledButImproperlyTraced1() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(dispatcher = false)", - " void scheduledButImproperlyTraced2() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(leaf = true)", - " void scheduledButImproperlyTraced3() {}", - "}") + """ + import com.newrelic.api.agent.Trace; + import org.springframework.scheduling.annotation.Scheduled; + + class A { + @Scheduled(fixedDelay = 1) + void scheduledButNotTraced() {} + + @Scheduled(fixedDelay = 1) + @Trace + void scheduledButImproperlyTraced1() {} + + @Scheduled(fixedDelay = 1) + @Trace(dispatcher = false) + void scheduledButImproperlyTraced2() {} + + @Scheduled(fixedDelay = 1) + @Trace(leaf = true) + void scheduledButImproperlyTraced3() {} + } + + """) .addOutputLines( "out/A.java", - "import com.newrelic.api.agent.Trace;", - "import org.springframework.scheduling.annotation.Scheduled;", - "", - "class A {", - " @Trace(dispatcher = true)", - " @Scheduled(fixedDelay = 1)", - " void scheduledButNotTraced() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(dispatcher = true)", - " void scheduledButImproperlyTraced1() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(dispatcher = true)", - " void scheduledButImproperlyTraced2() {}", - "", - " @Scheduled(fixedDelay = 1)", - " @Trace(dispatcher = true, leaf = true)", - " void scheduledButImproperlyTraced3() {}", - "}") + """ + import com.newrelic.api.agent.Trace; + import org.springframework.scheduling.annotation.Scheduled; + + class A { + @Trace(dispatcher = true) + @Scheduled(fixedDelay = 1) + void scheduledButNotTraced() {} + + @Scheduled(fixedDelay = 1) + @Trace(dispatcher = true) + void scheduledButImproperlyTraced1() {} + + @Scheduled(fixedDelay = 1) + @Trace(dispatcher = true) + void scheduledButImproperlyTraced2() {} + + @Scheduled(fixedDelay = 1) + @Trace(dispatcher = true, leaf = true) + void scheduledButImproperlyTraced3() {} + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/Slf4JLogStatementTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/Slf4JLogStatementTest.java index 5c9d98a7ad3..7706e0b7534 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/Slf4JLogStatementTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/Slf4JLogStatementTest.java @@ -16,80 +16,83 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "import org.slf4j.MarkerFactory;", - "", - "class A {", - " private static final String FMT0 = \"format-string-without-placeholders\";", - " private static final String FMT1 = \"format-string-with-{}-placeholder\";", - " private static final String FMT2 = \"format-string-with-{}-{}-placeholders\";", - " private static final String FMT_ERR = \"format-string-with-%s-placeholder\";", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " private final Marker marker = MarkerFactory.getMarker(A.class.getName());", - " private final Object o = new Object();", - " private final String s = o.toString();", - " private final Throwable t = new Throwable();", - "", - " void m() {", - " LOG.trace(s);", - " LOG.debug(s, o);", - " LOG.info(s, t);", - " LOG.warn(s, o, t);", - " LOG.error(marker, s);", - " LOG.trace(marker, s, o);", - " LOG.debug(marker, s, t);", - " LOG.info(marker, s, o, t);", - "", - " LOG.warn(FMT0);", - " // BUG: Diagnostic contains: Log statement contains 0 placeholders, but specifies 1 matching", - " // argument(s)", - " LOG.error(FMT0, o);", - " LOG.trace(FMT0, t);", - " // BUG: Diagnostic contains:", - " LOG.debug(FMT0, o, t);", - " LOG.info(marker, FMT0);", - " // BUG: Diagnostic contains:", - " LOG.warn(marker, FMT0, o);", - " LOG.error(marker, FMT0, t);", - " // BUG: Diagnostic contains:", - " LOG.trace(marker, FMT0, o, t);", - "", - " // BUG: Diagnostic contains: Log statement contains 1 placeholders, but specifies 0 matching", - " // argument(s)", - " LOG.debug(FMT1);", - " LOG.info(FMT1, o);", - " // BUG: Diagnostic contains:", - " LOG.warn(FMT1, t);", - " LOG.error(FMT1, o, t);", - " // BUG: Diagnostic contains: Log statement contains 1 placeholders, but specifies 2 matching", - " // argument(s)", - " LOG.trace(FMT1, o, o);", - " // BUG: Diagnostic contains:", - " LOG.debug(FMT1, o, o, t);", - " // BUG: Diagnostic contains:", - " LOG.info(marker, FMT1);", - " LOG.warn(marker, FMT1, o);", - " // BUG: Diagnostic contains:", - " LOG.error(marker, FMT1, t);", - " LOG.trace(marker, FMT1, o, t);", - " // BUG: Diagnostic contains:", - " LOG.debug(marker, FMT1, o, o);", - " // BUG: Diagnostic contains:", - " LOG.info(marker, FMT1, o, o, t);", - "", - " // BUG: Diagnostic contains: SLF4J log statement placeholders are of the form `{}`, not `%s`", - " LOG.warn(FMT_ERR);", - " // BUG: Diagnostic contains:", - " LOG.error(FMT_ERR, t);", - " // BUG: Diagnostic contains:", - " LOG.trace(FMT_ERR, o);", - " // BUG: Diagnostic contains:", - " LOG.debug(FMT_ERR, o, t);", - " }", - "}") + """ + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + import org.slf4j.MarkerFactory; + + class A { + private static final String FMT0 = "format-string-without-placeholders"; + private static final String FMT1 = "format-string-with-{}-placeholder"; + private static final String FMT2 = "format-string-with-{}-{}-placeholders"; + private static final String FMT_ERR = "format-string-with-%s-placeholder"; + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + private final Marker marker = MarkerFactory.getMarker(A.class.getName()); + private final Object o = new Object(); + private final String s = o.toString(); + private final Throwable t = new Throwable(); + + void m() { + LOG.trace(s); + LOG.debug(s, o); + LOG.info(s, t); + LOG.warn(s, o, t); + LOG.error(marker, s); + LOG.trace(marker, s, o); + LOG.debug(marker, s, t); + LOG.info(marker, s, o, t); + + LOG.warn(FMT0); + // BUG: Diagnostic contains: Log statement contains 0 placeholders, but specifies 1 matching + // argument(s) + LOG.error(FMT0, o); + LOG.trace(FMT0, t); + // BUG: Diagnostic contains: + LOG.debug(FMT0, o, t); + LOG.info(marker, FMT0); + // BUG: Diagnostic contains: + LOG.warn(marker, FMT0, o); + LOG.error(marker, FMT0, t); + // BUG: Diagnostic contains: + LOG.trace(marker, FMT0, o, t); + + // BUG: Diagnostic contains: Log statement contains 1 placeholders, but specifies 0 matching + // argument(s) + LOG.debug(FMT1); + LOG.info(FMT1, o); + // BUG: Diagnostic contains: + LOG.warn(FMT1, t); + LOG.error(FMT1, o, t); + // BUG: Diagnostic contains: Log statement contains 1 placeholders, but specifies 2 matching + // argument(s) + LOG.trace(FMT1, o, o); + // BUG: Diagnostic contains: + LOG.debug(FMT1, o, o, t); + // BUG: Diagnostic contains: + LOG.info(marker, FMT1); + LOG.warn(marker, FMT1, o); + // BUG: Diagnostic contains: + LOG.error(marker, FMT1, t); + LOG.trace(marker, FMT1, o, t); + // BUG: Diagnostic contains: + LOG.debug(marker, FMT1, o, o); + // BUG: Diagnostic contains: + LOG.info(marker, FMT1, o, o, t); + + // BUG: Diagnostic contains: SLF4J log statement placeholders are of the form `{}`, not `%s` + LOG.warn(FMT_ERR); + // BUG: Diagnostic contains: + LOG.error(FMT_ERR, t); + // BUG: Diagnostic contains: + LOG.trace(FMT_ERR, o); + // BUG: Diagnostic contains: + LOG.debug(FMT_ERR, o, t); + } + } + + """) .doTest(); } @@ -99,50 +102,56 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "import org.slf4j.MarkerFactory;", - "", - "class A {", - " private static final String FMT_ERR = \"format-string-with-%s-placeholder\";", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " private final Marker marker = MarkerFactory.getMarker(A.class.getName());", - " private final Object o = new Object();", - " private final String s = o.toString();", - " private final Throwable t = new Throwable();", - "", - " void m() {", - " LOG.error(FMT_ERR, o);", - " LOG.error(\"format-string-with-'%s'-placeholder\", o);", - " LOG.error(\"format-string-with-\\\"%s\\\"-placeholder\", o);", - " LOG.error(\"format-string-with-%s\" + \"-placeholder\", o);", - " }", - "}") + """ + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + import org.slf4j.MarkerFactory; + + class A { + private static final String FMT_ERR = "format-string-with-%s-placeholder"; + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + private final Marker marker = MarkerFactory.getMarker(A.class.getName()); + private final Object o = new Object(); + private final String s = o.toString(); + private final Throwable t = new Throwable(); + + void m() { + LOG.error(FMT_ERR, o); + LOG.error("format-string-with-'%s'-placeholder", o); + LOG.error("format-string-with-\\"%s\\"-placeholder", o); + LOG.error("format-string-with-%s" + "-placeholder", o); + } + } + + """) .addOutputLines( "out/A.java", - "import org.slf4j.Logger;", - "import org.slf4j.LoggerFactory;", - "import org.slf4j.Marker;", - "import org.slf4j.MarkerFactory;", - "", - "class A {", - " private static final String FMT_ERR = \"format-string-with-%s-placeholder\";", - " private static final Logger LOG = LoggerFactory.getLogger(A.class);", - "", - " private final Marker marker = MarkerFactory.getMarker(A.class.getName());", - " private final Object o = new Object();", - " private final String s = o.toString();", - " private final Throwable t = new Throwable();", - "", - " void m() {", - " LOG.error(FMT_ERR, o);", - " LOG.error(\"format-string-with-'{}'-placeholder\", o);", - " LOG.error(\"format-string-with-\\\"{}\\\"-placeholder\", o);", - " LOG.error(\"format-string-with-{}\" + \"-placeholder\", o);", - " }", - "}") + """ + import org.slf4j.Logger; + import org.slf4j.LoggerFactory; + import org.slf4j.Marker; + import org.slf4j.MarkerFactory; + + class A { + private static final String FMT_ERR = "format-string-with-%s-placeholder"; + private static final Logger LOG = LoggerFactory.getLogger(A.class); + + private final Marker marker = MarkerFactory.getMarker(A.class.getName()); + private final Object o = new Object(); + private final String s = o.toString(); + private final Throwable t = new Throwable(); + + void m() { + LOG.error(FMT_ERR, o); + LOG.error("format-string-with-'{}'-placeholder", o); + LOG.error("format-string-with-\\"{}\\"-placeholder", o); + LOG.error("format-string-with-{}" + "-placeholder", o); + } + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/SpringMvcAnnotationTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/SpringMvcAnnotationTest.java index 4527a86ca2e..acdcc245b03 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/SpringMvcAnnotationTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/SpringMvcAnnotationTest.java @@ -16,70 +16,73 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static org.springframework.web.bind.annotation.RequestMethod.DELETE;", - "import static org.springframework.web.bind.annotation.RequestMethod.GET;", - "import static org.springframework.web.bind.annotation.RequestMethod.HEAD;", - "import static org.springframework.web.bind.annotation.RequestMethod.PATCH;", - "import static org.springframework.web.bind.annotation.RequestMethod.POST;", - "import static org.springframework.web.bind.annotation.RequestMethod.PUT;", - "", - "import org.springframework.web.bind.annotation.DeleteMapping;", - "import org.springframework.web.bind.annotation.GetMapping;", - "import org.springframework.web.bind.annotation.PatchMapping;", - "import org.springframework.web.bind.annotation.PostMapping;", - "import org.springframework.web.bind.annotation.PutMapping;", - "import org.springframework.web.bind.annotation.RequestMapping;", - "import org.springframework.web.bind.annotation.RequestMethod;", - "", - "interface A {", - " @RequestMapping", - " A simple();", - "", - " @RequestMapping(method = {})", - " A explicitDefault();", - " // BUG: Diagnostic contains:", - " @RequestMapping(method = RequestMethod.GET)", - " A get();", - " // BUG: Diagnostic contains:", - " @RequestMapping(method = {RequestMethod.POST})", - " A post();", - " // BUG: Diagnostic contains:", - " @RequestMapping(method = {PUT})", - " A put();", - " // BUG: Diagnostic contains:", - " @RequestMapping(method = {DELETE})", - " A delete();", - " // BUG: Diagnostic contains:", - " @RequestMapping(method = {PATCH})", - " A patch();", - "", - " @RequestMapping(method = HEAD)", - " A head();", - "", - " @RequestMapping(method = RequestMethod.OPTIONS)", - " A options();", - "", - " @RequestMapping(method = {GET, POST})", - " A simpleMix();", - "", - " @RequestMapping(method = {RequestMethod.GET, RequestMethod.POST})", - " A verboseMix();", - "", - " @DeleteMapping", - " A properDelete();", - "", - " @GetMapping", - " A properGet();", - "", - " @PatchMapping", - " A properPatch();", - "", - " @PostMapping", - " A properPost();", - "", - " @PutMapping", - " A properPut();", - "}") + """ + import static org.springframework.web.bind.annotation.RequestMethod.DELETE; + import static org.springframework.web.bind.annotation.RequestMethod.GET; + import static org.springframework.web.bind.annotation.RequestMethod.HEAD; + import static org.springframework.web.bind.annotation.RequestMethod.PATCH; + import static org.springframework.web.bind.annotation.RequestMethod.POST; + import static org.springframework.web.bind.annotation.RequestMethod.PUT; + + import org.springframework.web.bind.annotation.DeleteMapping; + import org.springframework.web.bind.annotation.GetMapping; + import org.springframework.web.bind.annotation.PatchMapping; + import org.springframework.web.bind.annotation.PostMapping; + import org.springframework.web.bind.annotation.PutMapping; + import org.springframework.web.bind.annotation.RequestMapping; + import org.springframework.web.bind.annotation.RequestMethod; + + interface A { + @RequestMapping + A simple(); + + @RequestMapping(method = {}) + A explicitDefault(); + // BUG: Diagnostic contains: + @RequestMapping(method = RequestMethod.GET) + A get(); + // BUG: Diagnostic contains: + @RequestMapping(method = {RequestMethod.POST}) + A post(); + // BUG: Diagnostic contains: + @RequestMapping(method = {PUT}) + A put(); + // BUG: Diagnostic contains: + @RequestMapping(method = {DELETE}) + A delete(); + // BUG: Diagnostic contains: + @RequestMapping(method = {PATCH}) + A patch(); + + @RequestMapping(method = HEAD) + A head(); + + @RequestMapping(method = RequestMethod.OPTIONS) + A options(); + + @RequestMapping(method = {GET, POST}) + A simpleMix(); + + @RequestMapping(method = {RequestMethod.GET, RequestMethod.POST}) + A verboseMix(); + + @DeleteMapping + A properDelete(); + + @GetMapping + A properGet(); + + @PatchMapping + A properPatch(); + + @PostMapping + A properPost(); + + @PutMapping + A properPut(); + } + + """) .doTest(); } @@ -88,66 +91,72 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static org.springframework.web.bind.annotation.RequestMethod.PATCH;", - "import static org.springframework.web.bind.annotation.RequestMethod.POST;", - "import static org.springframework.web.bind.annotation.RequestMethod.PUT;", - "", - "import org.springframework.web.bind.annotation.RequestMapping;", - "import org.springframework.web.bind.annotation.RequestMethod;", - "", - "interface A {", - " @RequestMapping(method = RequestMethod.GET)", - " A simple();", - "", - " @RequestMapping(path = \"/foo/bar\", method = POST)", - " A prefixed();", - "", - " @RequestMapping(", - " method = {RequestMethod.DELETE},", - " path = \"/foo/bar\")", - " A suffixed();", - "", - " @RequestMapping(", - " path = \"/foo/bar\",", - " method = {PUT},", - " consumes = {\"a\", \"b\"})", - " A surrounded();", - "", - " @RequestMapping(method = {PATCH})", - " A curly();", - "}") + """ + import static org.springframework.web.bind.annotation.RequestMethod.PATCH; + import static org.springframework.web.bind.annotation.RequestMethod.POST; + import static org.springframework.web.bind.annotation.RequestMethod.PUT; + + import org.springframework.web.bind.annotation.RequestMapping; + import org.springframework.web.bind.annotation.RequestMethod; + + interface A { + @RequestMapping(method = RequestMethod.GET) + A simple(); + + @RequestMapping(path = "/foo/bar", method = POST) + A prefixed(); + + @RequestMapping( + method = {RequestMethod.DELETE}, + path = "/foo/bar") + A suffixed(); + + @RequestMapping( + path = "/foo/bar", + method = {PUT}, + consumes = {"a", "b"}) + A surrounded(); + + @RequestMapping(method = {PATCH}) + A curly(); + } + + """) .addOutputLines( "out/A.java", - "import static org.springframework.web.bind.annotation.RequestMethod.PATCH;", - "import static org.springframework.web.bind.annotation.RequestMethod.POST;", - "import static org.springframework.web.bind.annotation.RequestMethod.PUT;", - "", - "import org.springframework.web.bind.annotation.DeleteMapping;", - "import org.springframework.web.bind.annotation.GetMapping;", - "import org.springframework.web.bind.annotation.PatchMapping;", - "import org.springframework.web.bind.annotation.PostMapping;", - "import org.springframework.web.bind.annotation.PutMapping;", - "import org.springframework.web.bind.annotation.RequestMapping;", - "import org.springframework.web.bind.annotation.RequestMethod;", - "", - "interface A {", - " @GetMapping()", - " A simple();", - "", - " @PostMapping(path = \"/foo/bar\")", - " A prefixed();", - "", - " @DeleteMapping(path = \"/foo/bar\")", - " A suffixed();", - "", - " @PutMapping(", - " path = \"/foo/bar\",", - " consumes = {\"a\", \"b\"})", - " A surrounded();", - "", - " @PatchMapping()", - " A curly();", - "}") + """ + import static org.springframework.web.bind.annotation.RequestMethod.PATCH; + import static org.springframework.web.bind.annotation.RequestMethod.POST; + import static org.springframework.web.bind.annotation.RequestMethod.PUT; + + import org.springframework.web.bind.annotation.DeleteMapping; + import org.springframework.web.bind.annotation.GetMapping; + import org.springframework.web.bind.annotation.PatchMapping; + import org.springframework.web.bind.annotation.PostMapping; + import org.springframework.web.bind.annotation.PutMapping; + import org.springframework.web.bind.annotation.RequestMapping; + import org.springframework.web.bind.annotation.RequestMethod; + + interface A { + @GetMapping() + A simple(); + + @PostMapping(path = "/foo/bar") + A prefixed(); + + @DeleteMapping(path = "/foo/bar") + A suffixed(); + + @PutMapping( + path = "/foo/bar", + consumes = {"a", "b"}) + A surrounded(); + + @PatchMapping() + A curly(); + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/StaticImportTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/StaticImportTest.java index 76730bbcada..5b642fdfac0 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/StaticImportTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/StaticImportTest.java @@ -36,83 +36,86 @@ void identification() { compilationTestHelper .addSourceLines( "A.java", - "import static com.google.common.collect.ImmutableMap.toImmutableMap;", - "import static com.google.common.collect.ImmutableSet.toImmutableSet;", - "import static java.nio.charset.StandardCharsets.UTF_8;", - "import static java.util.function.Predicate.not;", - "import static org.springframework.boot.test.context.SpringBootTest.WebEnvironment.RANDOM_PORT;", - "", - "import com.google.common.base.Predicates;", - "import com.google.common.collect.ImmutableMap;", - "import com.google.common.collect.ImmutableMultiset;", - "import com.google.common.collect.ImmutableSet;", - "import com.google.errorprone.refaster.ImportPolicy;", - "import com.google.errorprone.refaster.annotation.UseImportPolicy;", - "import java.nio.charset.StandardCharsets;", - "import java.time.ZoneOffset;", - "import java.util.Optional;", - "import java.util.UUID;", - "import java.util.function.Predicate;", - "import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;", - "import org.springframework.http.MediaType;", - "", - "class A {", - " void m() {", - " // BUG: Diagnostic contains:", - " ImmutableMap.toImmutableMap(v -> v, v -> v);", - " ImmutableMap.toImmutableMap(v -> v, v -> v);", - " toImmutableMap(v -> v, v -> v);", - "", - " // BUG: Diagnostic contains:", - " ImmutableSet.toImmutableSet();", - " ImmutableSet.toImmutableSet();", - " toImmutableSet();", - "", - " // Not flagged because we define `#toImmutableMultiset` below.", - " ImmutableMultiset.toImmutableMultiset();", - " ImmutableMultiset.toImmutableMultiset();", - " toImmutableMultiset();", - "", - " // BUG: Diagnostic contains:", - " Predicate.not(null);", - " not(null);", - "", - " // BUG: Diagnostic contains:", - " Predicates.alwaysTrue();", - " // BUG: Diagnostic contains:", - " Predicates.alwaysFalse();", - " // Not flagged because of `java.util.function.Predicate.not` import.", - " Predicates.not(null);", - "", - " // BUG: Diagnostic contains:", - " UUID uuid = UUID.randomUUID();", - "", - " // BUG: Diagnostic contains:", - " Object o1 = StandardCharsets.UTF_8;", - " Object o2 = UTF_8;", - "", - " // BUG: Diagnostic contains:", - " Object e1 = WebEnvironment.RANDOM_PORT;", - " Object e2 = RANDOM_PORT;", - "", - " // Not flagged because `MediaType.ALL` is exempted.", - " MediaType t1 = MediaType.ALL;", - " // BUG: Diagnostic contains:", - " MediaType t2 = MediaType.APPLICATION_JSON;", - "", - " Optional.empty();", - "", - " // BUG: Diagnostic contains:", - " ZoneOffset zo1 = ZoneOffset.UTC;", - " ZoneOffset zo2 = ZoneOffset.MIN;", - " }", - "", - " // BUG: Diagnostic contains:", - " @UseImportPolicy(ImportPolicy.IMPORT_TOP_LEVEL)", - " void refasterAfterTemplate() {}", - "", - " void toImmutableMultiset() {}", - "}") + """ + import static com.google.common.collect.ImmutableMap.toImmutableMap; + import static com.google.common.collect.ImmutableSet.toImmutableSet; + import static java.nio.charset.StandardCharsets.UTF_8; + import static java.util.function.Predicate.not; + import static org.springframework.boot.test.context.SpringBootTest.WebEnvironment.RANDOM_PORT; + + import com.google.common.base.Predicates; + import com.google.common.collect.ImmutableMap; + import com.google.common.collect.ImmutableMultiset; + import com.google.common.collect.ImmutableSet; + import com.google.errorprone.refaster.ImportPolicy; + import com.google.errorprone.refaster.annotation.UseImportPolicy; + import java.nio.charset.StandardCharsets; + import java.time.ZoneOffset; + import java.util.Optional; + import java.util.UUID; + import java.util.function.Predicate; + import org.springframework.boot.test.context.SpringBootTest.WebEnvironment; + import org.springframework.http.MediaType; + + class A { + void m() { + // BUG: Diagnostic contains: + ImmutableMap.toImmutableMap(v -> v, v -> v); + ImmutableMap.toImmutableMap(v -> v, v -> v); + toImmutableMap(v -> v, v -> v); + + // BUG: Diagnostic contains: + ImmutableSet.toImmutableSet(); + ImmutableSet.toImmutableSet(); + toImmutableSet(); + + // Not flagged because we define `#toImmutableMultiset` below. + ImmutableMultiset.toImmutableMultiset(); + ImmutableMultiset.toImmutableMultiset(); + toImmutableMultiset(); + + // BUG: Diagnostic contains: + Predicate.not(null); + not(null); + + // BUG: Diagnostic contains: + Predicates.alwaysTrue(); + // BUG: Diagnostic contains: + Predicates.alwaysFalse(); + // Not flagged because of `java.util.function.Predicate.not` import. + Predicates.not(null); + + // BUG: Diagnostic contains: + UUID uuid = UUID.randomUUID(); + + // BUG: Diagnostic contains: + Object o1 = StandardCharsets.UTF_8; + Object o2 = UTF_8; + + // BUG: Diagnostic contains: + Object e1 = WebEnvironment.RANDOM_PORT; + Object e2 = RANDOM_PORT; + + // Not flagged because `MediaType.ALL` is exempted. + MediaType t1 = MediaType.ALL; + // BUG: Diagnostic contains: + MediaType t2 = MediaType.APPLICATION_JSON; + + Optional.empty(); + + // BUG: Diagnostic contains: + ZoneOffset zo1 = ZoneOffset.UTC; + ZoneOffset zo2 = ZoneOffset.MIN; + } + + // BUG: Diagnostic contains: + @UseImportPolicy(ImportPolicy.IMPORT_TOP_LEVEL) + void refasterAfterTemplate() {} + + void toImmutableMultiset() {} + } + + """) .doTest(); } @@ -121,154 +124,160 @@ void replacement() { refactoringTestHelper .addInputLines( "in/A.java", - "import static java.util.function.Predicate.not;", - "", - "import com.google.common.base.Predicates;", - "import com.google.common.collect.ImmutableMap;", - "import com.google.common.collect.ImmutableSet;", - "import com.google.errorprone.BugPattern;", - "import com.google.errorprone.BugPattern.SeverityLevel;", - "import java.nio.charset.StandardCharsets;", - "import java.util.ArrayList;", - "import java.util.Collections;", - "import java.util.Objects;", - "import java.util.regex.Pattern;", - "import org.junit.jupiter.params.provider.Arguments;", - "import org.springframework.boot.test.context.SpringBootTest;", - "import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;", - "import org.springframework.format.annotation.DateTimeFormat;", - "import org.springframework.format.annotation.DateTimeFormat.ISO;", - "import org.springframework.http.MediaType;", - "", - "class A {", - " void m1() {", - " ImmutableMap.toImmutableMap(v -> v, v -> v);", - " ImmutableMap.toImmutableMap(v -> v, v -> v);", - "", - " ImmutableSet.toImmutableSet();", - " ImmutableSet.toImmutableSet();", - "", - " Collections.disjoint(ImmutableSet.of(), ImmutableSet.of());", - " Collections.reverse(new ArrayList<>());", - "", - " Predicates.not(null);", - " not(null);", - "", - " Arguments.arguments(\"foo\");", - "", - " Objects.requireNonNull(\"bar\");", - "", - " Object o = StandardCharsets.UTF_8;", - "", - " ImmutableSet.of(", - " MediaType.ALL,", - " MediaType.APPLICATION_XHTML_XML,", - " MediaType.TEXT_HTML,", - " MediaType.valueOf(\"image/webp\"));", - "", - " Pattern.compile(\"\", Pattern.CASE_INSENSITIVE);", - " }", - "", - " void m2(", - " @DateTimeFormat(iso = DateTimeFormat.ISO.DATE) String date,", - " @DateTimeFormat(iso = DateTimeFormat.ISO.DATE_TIME) String dateTime,", - " @DateTimeFormat(iso = DateTimeFormat.ISO.TIME) String time) {}", - "", - " void m3(", - " @DateTimeFormat(iso = ISO.DATE) String date,", - " @DateTimeFormat(iso = ISO.DATE_TIME) String dateTime,", - " @DateTimeFormat(iso = ISO.TIME) String time) {}", - "", - " @BugPattern(", - " summary = \"\",", - " linkType = BugPattern.LinkType.NONE,", - " severity = SeverityLevel.SUGGESTION,", - " tags = BugPattern.StandardTags.SIMPLIFICATION)", - " static final class TestBugPattern {}", - "", - " @SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)", - " final class Test {}", - "}") + """ + import static java.util.function.Predicate.not; + + import com.google.common.base.Predicates; + import com.google.common.collect.ImmutableMap; + import com.google.common.collect.ImmutableSet; + import com.google.errorprone.BugPattern; + import com.google.errorprone.BugPattern.SeverityLevel; + import java.nio.charset.StandardCharsets; + import java.util.ArrayList; + import java.util.Collections; + import java.util.Objects; + import java.util.regex.Pattern; + import org.junit.jupiter.params.provider.Arguments; + import org.springframework.boot.test.context.SpringBootTest; + import org.springframework.boot.test.context.SpringBootTest.WebEnvironment; + import org.springframework.format.annotation.DateTimeFormat; + import org.springframework.format.annotation.DateTimeFormat.ISO; + import org.springframework.http.MediaType; + + class A { + void m1() { + ImmutableMap.toImmutableMap(v -> v, v -> v); + ImmutableMap.toImmutableMap(v -> v, v -> v); + + ImmutableSet.toImmutableSet(); + ImmutableSet.toImmutableSet(); + + Collections.disjoint(ImmutableSet.of(), ImmutableSet.of()); + Collections.reverse(new ArrayList<>()); + + Predicates.not(null); + not(null); + + Arguments.arguments("foo"); + + Objects.requireNonNull("bar"); + + Object o = StandardCharsets.UTF_8; + + ImmutableSet.of( + MediaType.ALL, + MediaType.APPLICATION_XHTML_XML, + MediaType.TEXT_HTML, + MediaType.valueOf("image/webp")); + + Pattern.compile("", Pattern.CASE_INSENSITIVE); + } + + void m2( + @DateTimeFormat(iso = DateTimeFormat.ISO.DATE) String date, + @DateTimeFormat(iso = DateTimeFormat.ISO.DATE_TIME) String dateTime, + @DateTimeFormat(iso = DateTimeFormat.ISO.TIME) String time) {} + + void m3( + @DateTimeFormat(iso = ISO.DATE) String date, + @DateTimeFormat(iso = ISO.DATE_TIME) String dateTime, + @DateTimeFormat(iso = ISO.TIME) String time) {} + + @BugPattern( + summary = "", + linkType = BugPattern.LinkType.NONE, + severity = SeverityLevel.SUGGESTION, + tags = BugPattern.StandardTags.SIMPLIFICATION) + static final class TestBugPattern {} + + @SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT) + final class Test {} + } + + """) .addOutputLines( "out/A.java", - "import static com.google.common.collect.ImmutableMap.toImmutableMap;", - "import static com.google.common.collect.ImmutableSet.toImmutableSet;", - "import static com.google.errorprone.BugPattern.LinkType.NONE;", - "import static com.google.errorprone.BugPattern.SeverityLevel.SUGGESTION;", - "import static com.google.errorprone.BugPattern.StandardTags.SIMPLIFICATION;", - "import static java.nio.charset.StandardCharsets.UTF_8;", - "import static java.util.Collections.disjoint;", - "import static java.util.Collections.reverse;", - "import static java.util.Objects.requireNonNull;", - "import static java.util.function.Predicate.not;", - "import static java.util.regex.Pattern.CASE_INSENSITIVE;", - "import static org.junit.jupiter.params.provider.Arguments.arguments;", - "import static org.springframework.boot.test.context.SpringBootTest.WebEnvironment.RANDOM_PORT;", - "import static org.springframework.format.annotation.DateTimeFormat.ISO.DATE;", - "import static org.springframework.format.annotation.DateTimeFormat.ISO.DATE_TIME;", - "import static org.springframework.format.annotation.DateTimeFormat.ISO.TIME;", - "import static org.springframework.http.MediaType.APPLICATION_XHTML_XML;", - "import static org.springframework.http.MediaType.TEXT_HTML;", - "", - "import com.google.common.base.Predicates;", - "import com.google.common.collect.ImmutableMap;", - "import com.google.common.collect.ImmutableSet;", - "import com.google.errorprone.BugPattern;", - "import com.google.errorprone.BugPattern.SeverityLevel;", - "import java.nio.charset.StandardCharsets;", - "import java.util.ArrayList;", - "import java.util.Collections;", - "import java.util.Objects;", - "import java.util.regex.Pattern;", - "import org.junit.jupiter.params.provider.Arguments;", - "import org.springframework.boot.test.context.SpringBootTest;", - "import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;", - "import org.springframework.format.annotation.DateTimeFormat;", - "import org.springframework.format.annotation.DateTimeFormat.ISO;", - "import org.springframework.http.MediaType;", - "", - "class A {", - " void m1() {", - " toImmutableMap(v -> v, v -> v);", - " ImmutableMap.toImmutableMap(v -> v, v -> v);", - "", - " toImmutableSet();", - " ImmutableSet.toImmutableSet();", - "", - " disjoint(ImmutableSet.of(), ImmutableSet.of());", - " reverse(new ArrayList<>());", - "", - " Predicates.not(null);", - " not(null);", - "", - " arguments(\"foo\");", - "", - " requireNonNull(\"bar\");", - "", - " Object o = UTF_8;", - "", - " ImmutableSet.of(", - " MediaType.ALL, APPLICATION_XHTML_XML, TEXT_HTML, MediaType.valueOf(\"image/webp\"));", - "", - " Pattern.compile(\"\", CASE_INSENSITIVE);", - " }", - "", - " void m2(", - " @DateTimeFormat(iso = DATE) String date,", - " @DateTimeFormat(iso = DATE_TIME) String dateTime,", - " @DateTimeFormat(iso = TIME) String time) {}", - "", - " void m3(", - " @DateTimeFormat(iso = DATE) String date,", - " @DateTimeFormat(iso = DATE_TIME) String dateTime,", - " @DateTimeFormat(iso = TIME) String time) {}", - "", - " @BugPattern(summary = \"\", linkType = NONE, severity = SUGGESTION, tags = SIMPLIFICATION)", - " static final class TestBugPattern {}", - "", - " @SpringBootTest(webEnvironment = RANDOM_PORT)", - " final class Test {}", - "}") + """ + import static com.google.common.collect.ImmutableMap.toImmutableMap; + import static com.google.common.collect.ImmutableSet.toImmutableSet; + import static com.google.errorprone.BugPattern.LinkType.NONE; + import static com.google.errorprone.BugPattern.SeverityLevel.SUGGESTION; + import static com.google.errorprone.BugPattern.StandardTags.SIMPLIFICATION; + import static java.nio.charset.StandardCharsets.UTF_8; + import static java.util.Collections.disjoint; + import static java.util.Collections.reverse; + import static java.util.Objects.requireNonNull; + import static java.util.function.Predicate.not; + import static java.util.regex.Pattern.CASE_INSENSITIVE; + import static org.junit.jupiter.params.provider.Arguments.arguments; + import static org.springframework.boot.test.context.SpringBootTest.WebEnvironment.RANDOM_PORT; + import static org.springframework.format.annotation.DateTimeFormat.ISO.DATE; + import static org.springframework.format.annotation.DateTimeFormat.ISO.DATE_TIME; + import static org.springframework.format.annotation.DateTimeFormat.ISO.TIME; + import static org.springframework.http.MediaType.APPLICATION_XHTML_XML; + import static org.springframework.http.MediaType.TEXT_HTML; + + import com.google.common.base.Predicates; + import com.google.common.collect.ImmutableMap; + import com.google.common.collect.ImmutableSet; + import com.google.errorprone.BugPattern; + import com.google.errorprone.BugPattern.SeverityLevel; + import java.nio.charset.StandardCharsets; + import java.util.ArrayList; + import java.util.Collections; + import java.util.Objects; + import java.util.regex.Pattern; + import org.junit.jupiter.params.provider.Arguments; + import org.springframework.boot.test.context.SpringBootTest; + import org.springframework.boot.test.context.SpringBootTest.WebEnvironment; + import org.springframework.format.annotation.DateTimeFormat; + import org.springframework.format.annotation.DateTimeFormat.ISO; + import org.springframework.http.MediaType; + + class A { + void m1() { + toImmutableMap(v -> v, v -> v); + ImmutableMap.toImmutableMap(v -> v, v -> v); + + toImmutableSet(); + ImmutableSet.toImmutableSet(); + + disjoint(ImmutableSet.of(), ImmutableSet.of()); + reverse(new ArrayList<>()); + + Predicates.not(null); + not(null); + + arguments("foo"); + + requireNonNull("bar"); + + Object o = UTF_8; + + ImmutableSet.of( + MediaType.ALL, APPLICATION_XHTML_XML, TEXT_HTML, MediaType.valueOf("image/webp")); + + Pattern.compile("", CASE_INSENSITIVE); + } + + void m2( + @DateTimeFormat(iso = DATE) String date, + @DateTimeFormat(iso = DATE_TIME) String dateTime, + @DateTimeFormat(iso = TIME) String time) {} + + void m3( + @DateTimeFormat(iso = DATE) String date, + @DateTimeFormat(iso = DATE_TIME) String dateTime, + @DateTimeFormat(iso = TIME) String time) {} + + @BugPattern(summary = "", linkType = NONE, severity = SUGGESTION, tags = SIMPLIFICATION) + static final class TestBugPattern {} + + @SpringBootTest(webEnvironment = RANDOM_PORT) + final class Test {} + } + + """) .doTest(TestMode.TEXT_MATCH); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/TimeZoneUsageTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/TimeZoneUsageTest.java index b675225b925..159b26156dc 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/TimeZoneUsageTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/TimeZoneUsageTest.java @@ -18,86 +18,89 @@ void identification() { compilationHelper .addSourceLines( "A.java", - "import static java.time.ZoneOffset.UTC;", - "", - "import java.time.Clock;", - "import java.time.Duration;", - "import java.time.Instant;", - "import java.time.LocalDate;", - "import java.time.LocalDateTime;", - "import java.time.LocalTime;", - "import java.time.ZoneId;", - "", - "class A {", - " void m() {", - " Clock clock = Clock.fixed(Instant.EPOCH, UTC);", - " clock.instant();", - " clock.millis();", - " Clock.offset(clock, Duration.ZERO);", - " Clock.tick(clock, Duration.ZERO);", - "", - " // BUG: Diagnostic matches: X", - " Clock.systemUTC();", - " // BUG: Diagnostic matches: X", - " Clock.systemDefaultZone();", - " // BUG: Diagnostic matches: X", - " Clock.system(UTC);", - " // BUG: Diagnostic matches: X", - " Clock.tickMillis(UTC);", - " // BUG: Diagnostic matches: X", - " Clock.tickMinutes(UTC);", - " // BUG: Diagnostic matches: X", - " Clock.tickSeconds(UTC);", - " // BUG: Diagnostic matches: X", - " clock.getZone();", - " // BUG: Diagnostic matches: X", - " clock.withZone(UTC);", - "", - " // BUG: Diagnostic matches: X", - " Instant.now();", - " // This is equivalent to `clock.instant()`, which is fine.", - " Instant.now(clock);", - "", - " // BUG: Diagnostic matches: X", - " LocalDate.now();", - " // BUG: Diagnostic matches: X", - " LocalDate.now(clock);", - " // BUG: Diagnostic matches: X", - " LocalDate.now(UTC);", - "", - " // BUG: Diagnostic matches: X", - " LocalDateTime.now();", - " // BUG: Diagnostic matches: X", - " LocalDateTime.now(clock);", - " // BUG: Diagnostic matches: X", - " LocalDateTime.now(UTC);", - "", - " // BUG: Diagnostic matches: X", - " LocalTime.now();", - " // BUG: Diagnostic matches: X", - " LocalTime.now(clock);", - " // BUG: Diagnostic matches: X", - " LocalTime.now(UTC);", - " }", - "", - " abstract class ForwardingClock extends Clock {", - " private final Clock clock;", - "", - " ForwardingClock(Clock clock) {", - " this.clock = clock;", - " }", - "", - " @Override", - " public ZoneId getZone() {", - " return clock.getZone();", - " }", - "", - " @Override", - " public Clock withZone(ZoneId zone) {", - " return clock.withZone(zone);", - " }", - " }", - "}") + """ + import static java.time.ZoneOffset.UTC; + + import java.time.Clock; + import java.time.Duration; + import java.time.Instant; + import java.time.LocalDate; + import java.time.LocalDateTime; + import java.time.LocalTime; + import java.time.ZoneId; + + class A { + void m() { + Clock clock = Clock.fixed(Instant.EPOCH, UTC); + clock.instant(); + clock.millis(); + Clock.offset(clock, Duration.ZERO); + Clock.tick(clock, Duration.ZERO); + + // BUG: Diagnostic matches: X + Clock.systemUTC(); + // BUG: Diagnostic matches: X + Clock.systemDefaultZone(); + // BUG: Diagnostic matches: X + Clock.system(UTC); + // BUG: Diagnostic matches: X + Clock.tickMillis(UTC); + // BUG: Diagnostic matches: X + Clock.tickMinutes(UTC); + // BUG: Diagnostic matches: X + Clock.tickSeconds(UTC); + // BUG: Diagnostic matches: X + clock.getZone(); + // BUG: Diagnostic matches: X + clock.withZone(UTC); + + // BUG: Diagnostic matches: X + Instant.now(); + // This is equivalent to `clock.instant()`, which is fine. + Instant.now(clock); + + // BUG: Diagnostic matches: X + LocalDate.now(); + // BUG: Diagnostic matches: X + LocalDate.now(clock); + // BUG: Diagnostic matches: X + LocalDate.now(UTC); + + // BUG: Diagnostic matches: X + LocalDateTime.now(); + // BUG: Diagnostic matches: X + LocalDateTime.now(clock); + // BUG: Diagnostic matches: X + LocalDateTime.now(UTC); + + // BUG: Diagnostic matches: X + LocalTime.now(); + // BUG: Diagnostic matches: X + LocalTime.now(clock); + // BUG: Diagnostic matches: X + LocalTime.now(UTC); + } + + abstract class ForwardingClock extends Clock { + private final Clock clock; + + ForwardingClock(Clock clock) { + this.clock = clock; + } + + @Override + public ZoneId getZone() { + return clock.getZone(); + } + + @Override + public Clock withZone(ZoneId zone) { + return clock.withZone(zone); + } + } + } + + """) .doTest(); } } diff --git a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/util/MethodMatcherFactoryTest.java b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/util/MethodMatcherFactoryTest.java index 7bd24f2f9a5..17f42b6ca2a 100644 --- a/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/util/MethodMatcherFactoryTest.java +++ b/error-prone-contrib/src/test/java/tech/picnic/errorprone/bugpatterns/util/MethodMatcherFactoryTest.java @@ -61,153 +61,171 @@ void matcher() { compilationTestHelper .addSourceLines( "com/example/A.java", - "package com.example;", - "", - "public class A {", - " public void m1() {}", - "", - " public void m1(String s) {}", - "", - " public void m1(int i, int j) {}", - "", - " public void m2() {}", - "", - " public void m2(String s) {}", - "", - " public void m2(int i, int j) {}", - "", - " public void m3() {}", - "", - " public void m3(String s) {}", - "", - " public void m3(int i, int j) {}", - "}") + """ + package com.example; + + public class A { + public void m1() {} + + public void m1(String s) {} + + public void m1(int i, int j) {} + + public void m2() {} + + public void m2(String s) {} + + public void m2(int i, int j) {} + + public void m3() {} + + public void m3(String s) {} + + public void m3(int i, int j) {} + } + + """) .addSourceLines( "com/example/B.java", - "package com.example;", - "", - "public class B {", - " public void m1() {}", - "", - " public void m1(String s) {}", - "", - " public void m1(int i, int j) {}", - "", - " public void m2() {}", - "", - " public void m2(String s) {}", - "", - " public void m2(int i, int j) {}", - "", - " public void m3() {}", - "", - " public void m3(String s) {}", - "", - " public void m3(int i, int j) {}", - "}") + """ + package com.example; + + public class B { + public void m1() {} + + public void m1(String s) {} + + public void m1(int i, int j) {} + + public void m2() {} + + public void m2(String s) {} + + public void m2(int i, int j) {} + + public void m3() {} + + public void m3(String s) {} + + public void m3(int i, int j) {} + } + + """) .addSourceLines( "com/example/sub/A.java", - "package com.example.sub;", - "", - "public class A {", - " public static void m1() {}", - "", - " public static void m1(String s) {}", - "", - " public static void m1(int i, int j) {}", - "", - " public static void m2() {}", - "", - " public static void m2(String s) {}", - "", - " public static void m2(int i, int j) {}", - "", - " public static void m3() {}", - "", - " public static void m3(String s) {}", - "", - " public static void m3(int i, int j) {}", - "}") + """ + package com.example.sub; + + public class A { + public static void m1() {} + + public static void m1(String s) {} + + public static void m1(int i, int j) {} + + public static void m2() {} + + public static void m2(String s) {} + + public static void m2(int i, int j) {} + + public static void m3() {} + + public static void m3(String s) {} + + public static void m3(int i, int j) {} + } + + """) .addSourceLines( "com/example/sub/B.java", - "package com.example.sub;", - "", - "public class B {", - " public static void m1() {}", - "", - " public static void m1(String s) {}", - "", - " public static void m1(int i, int j) {}", - "", - " public static void m2() {}", - "", - " public static void m2(String s) {}", - "", - " public static void m2(int i, int j) {}", - "", - " public static void m3() {}", - "", - " public static void m3(String s) {}", - "", - " public static void m3(int i, int j) {}", - "}") + """ + package com.example.sub; + + public class B { + public static void m1() {} + + public static void m1(String s) {} + + public static void m1(int i, int j) {} + + public static void m2() {} + + public static void m2(String s) {} + + public static void m2(int i, int j) {} + + public static void m3() {} + + public static void m3(String s) {} + + public static void m3(int i, int j) {} + } + + """) .addSourceLines( "External.java", - "import com.example.A;", - "import com.example.sub.B;", - "", - "public class External {", - " void invocations() {", - " // BUG: Diagnostic contains:", - " new A().m1();", - " new A().m1(\"\");", - " new A().m1(0, 0);", - " new A().m2();", - " // BUG: Diagnostic contains:", - " new A().m2(\"\");", - " new A().m2(0, 0);", - " new A().m3();", - " new A().m3(\"\");", - " new A().m3(0, 0);", - " B.m1();", - " B.m1(\"\");", - " B.m1(0, 0);", - " B.m2();", - " B.m2(\"\");", - " B.m2(0, 0);", - " B.m3();", - " B.m3(\"\");", - " // BUG: Diagnostic contains:", - " B.m3(0, 0);", - " }", - "}") + """ + import com.example.A; + import com.example.sub.B; + + public class External { + void invocations() { + // BUG: Diagnostic contains: + new A().m1(); + new A().m1(""); + new A().m1(0, 0); + new A().m2(); + // BUG: Diagnostic contains: + new A().m2(""); + new A().m2(0, 0); + new A().m3(); + new A().m3(""); + new A().m3(0, 0); + B.m1(); + B.m1(""); + B.m1(0, 0); + B.m2(); + B.m2(""); + B.m2(0, 0); + B.m3(); + B.m3(""); + // BUG: Diagnostic contains: + B.m3(0, 0); + } + } + + """) .addSourceLines( "ExternalWithDifferentPackages.java", - "import com.example.B;", - "import com.example.sub.A;", - "", - "public class ExternalWithDifferentPackages {", - " void invocations() {", - " A.m1();", - " A.m1(\"\");", - " A.m1(0, 0);", - " A.m2();", - " A.m2(\"\");", - " A.m2(0, 0);", - " A.m3();", - " A.m3(\"\");", - " A.m3(0, 0);", - " new B().m1();", - " new B().m1(\"\");", - " new B().m1(0, 0);", - " new B().m2();", - " new B().m2(\"\");", - " new B().m2(0, 0);", - " new B().m3();", - " new B().m3(\"\");", - " new B().m3(0, 0);", - " }", - "}") + """ + import com.example.B; + import com.example.sub.A; + + public class ExternalWithDifferentPackages { + void invocations() { + A.m1(); + A.m1(""); + A.m1(0, 0); + A.m2(); + A.m2(""); + A.m2(0, 0); + A.m3(); + A.m3(""); + A.m3(0, 0); + new B().m1(); + new B().m1(""); + new B().m1(0, 0); + new B().m2(); + new B().m2(""); + new B().m2(0, 0); + new B().m3(); + new B().m3(""); + new B().m3(0, 0); + } + } + + """) .doTest(); } } diff --git a/refaster-support/src/test/java/tech/picnic/errorprone/refaster/util/IsArrayTest.java b/refaster-support/src/test/java/tech/picnic/errorprone/refaster/util/IsArrayTest.java index 11ad12c4620..b3c10373e08 100644 --- a/refaster-support/src/test/java/tech/picnic/errorprone/refaster/util/IsArrayTest.java +++ b/refaster-support/src/test/java/tech/picnic/errorprone/refaster/util/IsArrayTest.java @@ -17,38 +17,41 @@ void matches() { CompilationTestHelper.newInstance(TestChecker.class, getClass()) .addSourceLines( "A.java", - "class A {", - " Object negative1() {", - " return alwaysNull();", - " }", - "", - " String negative2() {", - " return alwaysNull();", - " }", - "", - " int negative3() {", - " return alwaysNull();", - " }", - "", - " Object[] positive1() {", - " // BUG: Diagnostic contains:", - " return alwaysNull();", - " }", - "", - " String[] positive2() {", - " // BUG: Diagnostic contains:", - " return alwaysNull();", - " }", - "", - " int[] positive3() {", - " // BUG: Diagnostic contains:", - " return alwaysNull();", - " }", - "", - " private static T alwaysNull() {", - " return null;", - " }", - "}") + """ + class A { + Object negative1() { + return alwaysNull(); + } + + String negative2() { + return alwaysNull(); + } + + int negative3() { + return alwaysNull(); + } + + Object[] positive1() { + // BUG: Diagnostic contains: + return alwaysNull(); + } + + String[] positive2() { + // BUG: Diagnostic contains: + return alwaysNull(); + } + + int[] positive3() { + // BUG: Diagnostic contains: + return alwaysNull(); + } + + private static T alwaysNull() { + return null; + } + } + + """) .doTest(); }