From 875cf56f534f9066219fe87d34aba0704f3d813f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?L=C3=A1zaro=20Clapp?= Date: Mon, 22 Feb 2021 14:45:41 -0500 Subject: [PATCH] [ContractHandler] Handle @Contract with empty value. (#450) The IntelliJ `@Contract` annotation has two fields, `value` and `pure`. When `pure = true`, `value` can be left empty. As such, the following is a valid `@Contract` annotation: ``` @Contract( pure = true ) ``` See. https://www.jetbrains.com/help/idea/contract-annotations.html Here, retriving the annotation value produces an empty string. However, existing ContractHandler logic expects `value` to either be `null` or a properly formated string in IntelliJ contract syntax (`"[...] -> [...]; [...]"`). Previous to this change, NullAway would report an error AND crash when seeing the `@Contract` annotation above, rather than ignore it. This patch fixes this. --- .../handlers/contract/ContractHandler.java | 2 +- .../java/com/uber/nullaway/NullAwayTest.java | 40 +++++++++++++++++++ 2 files changed, 41 insertions(+), 1 deletion(-) diff --git a/nullaway/src/main/java/com/uber/nullaway/handlers/contract/ContractHandler.java b/nullaway/src/main/java/com/uber/nullaway/handlers/contract/ContractHandler.java index 3c8c48a3e2..f89455f79a 100644 --- a/nullaway/src/main/java/com/uber/nullaway/handlers/contract/ContractHandler.java +++ b/nullaway/src/main/java/com/uber/nullaway/handlers/contract/ContractHandler.java @@ -101,7 +101,7 @@ public NullnessHint onDataflowVisitMethodInvocation( Preconditions.checkNotNull(callee); // Check to see if this method has an @Contract annotation String contractString = NullabilityUtil.getAnnotationValue(callee, CONTRACT_ANNOTATION_NAME); - if (contractString != null) { + if (contractString != null && contractString.trim().length() > 0) { // Found a contract, lets parse it. String[] clauses = contractString.split(";"); for (String clause : clauses) { diff --git a/nullaway/src/test/java/com/uber/nullaway/NullAwayTest.java b/nullaway/src/test/java/com/uber/nullaway/NullAwayTest.java index 8860177150..dd099528a4 100644 --- a/nullaway/src/test/java/com/uber/nullaway/NullAwayTest.java +++ b/nullaway/src/test/java/com/uber/nullaway/NullAwayTest.java @@ -566,6 +566,46 @@ public void contractNonVarArg() { .doTest(); } + @Test + public void contractPureOnlyIgnored() { + compilationHelper + .setArgs( + Arrays.asList( + "-d", + temporaryFolder.getRoot().getAbsolutePath(), + "-XepOpt:NullAway:AnnotatedPackages=com.uber")) + .addSourceLines( + "PureLibrary.java", + "package com.example.library;", + "import org.jetbrains.annotations.Contract;", + "public class PureLibrary {", + " @Contract(", + " pure = true", + " )", + " public static String pi() {", + " // Meh, close enough...", + " return Double.toString(3.14);", + " }", + "}") + .addSourceLines( + "Test.java", + "package com.uber;", + "import com.example.library.PureLibrary;", + "import javax.annotation.Nullable;", + "class Test {", + " String piValue() {", + " String pi = PureLibrary.pi();", + " // Note: we must trigger dataflow to ensure that", + " // ContractHandler.onDataflowVisitMethodInvocation is called", + " if (pi != null) {", + " return pi;", + " }", + " return Integer.toString(3);", + " }", + "}") + .doTest(); + } + @Test public void testEnumInit() { compilationHelper