diff --git a/README.md b/README.md index b2df9f3..b951aed 100644 --- a/README.md +++ b/README.md @@ -258,6 +258,54 @@ You can toggle the various inspections in the Settings/Editor/Inspections in the choosing between both fixes (does not work well for batch processing), or ignore this edge case altogether (just to be sure to not break any code). +- AssertThatFileExpression + + Moves File method calls inside ```assertThat()``` out. + + ``` + from: assertThat(file.canRead()).isTrue(); + to: assertThat(file).canRead(); + + from: assertThat(file.canWrite()).isTrue(); + to: assertThat(file).canWrite(); + + from: assertThat(file.exists()).isTrue(); + to: assertThat(file).exists(); + + from: assertThat(file.exists()).isFalse(); + to: assertThat(file).doesNotExist(); + + from: assertThat(file.isAbsolute()).isTrue(); + to: assertThat(file).isAbsolute(); + + from: assertThat(file.isAbsolute()).isFalse(); + to: assertThat(file).isRelative(); + + from: assertThat(file.isDirectory()).isTrue(); + to: assertThat(file).isDirectory(); + + from: assertThat(file.isFile()).isTrue(); + to: assertThat(file).isFile(); + + from: assertThat(file.getName()).isEqualTo(filename); + to: assertThat(file).hasName(filename); + + from: assertThat(file.getParent()).isEqualTo(pathname); + to: assertThat(file).hasParent(pathname); + + from: assertThat(file.getParent()).isNull(); + from: assertThat(file.getParentFile()).isNull(); + to: assertThat(file).hasNoParent(); + + from: assertThat(file.list()).isEmpty(); + from: assertThat(file.listFiles()).isEmpty(); + to: assertThat(file).isEmptyDirectory(); + + from: assertThat(file.list()).isNotEmpty(); + from: assertThat(file.listFiles()).isNotEmpty(); + to: assertThat(file).isNotEmptyDirectory(); + ``` + - AssertThatEnumerableIsEmpty Uses ```isEmpty()``` for ```hasSize(0)``` iterable assertions instead. @@ -552,10 +600,10 @@ The IntelliJ framework actually uses the JUnit 3 TestCase for plugin testing and Feel free to use the code (in package ```de.platon42.intellij.jupiter```) for your projects (with attribution). ## Planned features -- More Optional fixes such as opt1.get() == opt2.get() etc. -- More moving out of methods for File, Path, LocalDate/Time etc. +- More Optional fixes such as ```opt1.get() == opt2.get()``` etc. +- More moving out of methods for Path, LocalDate/Time etc. - Converting ```foo.compareTo(bar) == 0``` to ```isEqualTo()``` (yes, I've *really* seen code like that) -- Extraction with property names to lambda with Java 8 +- Extraction with property names to lambda/method reference with Java 8 ``` from: assertThat(object).extracting("propOne", "propNoGetter", "propTwo.innerProp")... @@ -569,6 +617,8 @@ Feel free to use the code (in package ```de.platon42.intellij.jupiter```) for yo for array types. Sigh. Shouldn't be working >12h a day and then do some more stuff at home. - Fixed a bug in AssertThatBinaryExpression inspection for ```assertThat(null != expression)``` and related that would not correctly invert the condition on transformation. +- Added new AssertThatFileExpression to move out many common methods from inside the + ```assertThat()``` expression (```exists(), getName(), getParent()```, and many more). #### V1.5 (24-Sep-19) - Fix for AssertThatCollectionOrMap inspection sometimes causing an index out of bounds exception. diff --git a/build.gradle b/build.gradle index 9a0f55b..647c3ab 100644 --- a/build.gradle +++ b/build.gradle @@ -49,6 +49,8 @@ patchPluginXml { for array types. Sigh. Shouldn't be working >12h a day and then do some more stuff at home.
Full changelog available at Github project site.
""" diff --git a/src/main/java/de/platon42/intellij/plugins/cajon/AssertJClassNames.kt b/src/main/java/de/platon42/intellij/plugins/cajon/AssertJClassNames.kt index 1d306d0..3b93318 100644 --- a/src/main/java/de/platon42/intellij/plugins/cajon/AssertJClassNames.kt +++ b/src/main/java/de/platon42/intellij/plugins/cajon/AssertJClassNames.kt @@ -44,6 +44,8 @@ class AssertJClassNames { @NonNls const val ABSTRACT_MAP_ASSERT_CLASSNAME = "org.assertj.core.api.AbstractMapAssert" @NonNls + const val ABSTRACT_OBJECT_ARRAY_ASSERT_CLASSNAME = "org.assertj.core.api.AbstractObjectArrayAssert" + @NonNls const val ABSTRACT_ITERABLE_ASSERT_CLASSNAME = "org.assertj.core.api.AbstractIterableAssert" @NonNls const val ABSTRACT_OPTIONAL_ASSERT_CLASSNAME = "org.assertj.core.api.AbstractOptionalAssert" diff --git a/src/main/java/de/platon42/intellij/plugins/cajon/inspections/AssertThatFileExpressionInspection.kt b/src/main/java/de/platon42/intellij/plugins/cajon/inspections/AssertThatFileExpressionInspection.kt new file mode 100644 index 0000000..0e2bc95 --- /dev/null +++ b/src/main/java/de/platon42/intellij/plugins/cajon/inspections/AssertThatFileExpressionInspection.kt @@ -0,0 +1,117 @@ +package de.platon42.intellij.plugins.cajon.inspections + +import com.intellij.codeInspection.ProblemsHolder +import com.intellij.psi.* +import com.siyeh.ig.callMatcher.CallMatcher +import de.platon42.intellij.plugins.cajon.* +import de.platon42.intellij.plugins.cajon.quickfixes.MoveOutMethodCallExpressionQuickFix + + +class AssertThatFileExpressionInspection : AbstractAssertJInspection() { + + companion object { + private const val DISPLAY_NAME = "Asserting a file specific expression" + + private val MAPPINGS = listOf( + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "canRead"), + "canRead", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "canWrite"), + "canWrite", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "exists"), + "exists", "doesNotExist", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "isAbsolute"), + "isAbsolute", "isRelative", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "isDirectory"), + "isDirectory", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "isFile"), + "isFile", expectBoolean = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "getName"), + "hasName", + expectedMatcher = CallMatcher.anyOf(IS_EQUAL_TO_OBJECT, IS_EQUAL_TO_STRING) + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "getParent", "getParentFile"), + "hasNoParent", expectNullNonNull = true + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "getParent"), + "hasParent", + expectedMatcher = CallMatcher.anyOf(IS_EQUAL_TO_OBJECT, IS_EQUAL_TO_STRING) + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "getParentFile"), + "hasParent", + expectedMatcher = IS_EQUAL_TO_OBJECT + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "list", "listFiles"), + "isEmptyDirectory", + expectedMatcher = CallMatcher.instanceCall(AssertJClassNames.ABSTRACT_OBJECT_ARRAY_ASSERT_CLASSNAME, MethodNames.IS_EMPTY) + .parameterCount(0)!! + ), + Mapping( + CallMatcher.instanceCall(CommonClassNames.JAVA_IO_FILE, "list", "listFiles"), + "isNotEmptyDirectory", + expectedMatcher = CallMatcher.instanceCall(AssertJClassNames.ABSTRACT_OBJECT_ARRAY_ASSERT_CLASSNAME, MethodNames.IS_NOT_EMPTY) + .parameterCount(0)!! + ) + ) + } + + override fun getDisplayName() = DISPLAY_NAME + + override fun buildVisitor(holder: ProblemsHolder, isOnTheFly: Boolean): PsiElementVisitor { + return object : JavaElementVisitor() { + override fun visitExpressionStatement(statement: PsiExpressionStatement) { + super.visitExpressionStatement(statement) + if (!statement.hasAssertThat()) return + val staticMethodCall = statement.findStaticMethodCall() ?: return + + val assertThatArgument = staticMethodCall.getArgOrNull(0) as? PsiMethodCallExpression ?: return + val expectedCallExpression = statement.findOutmostMethodCall() ?: return + + for (mapping in MAPPINGS.filter { it.callMatcher.test(assertThatArgument) }) { + if (mapping.expectBoolean && ASSERT_THAT_BOOLEAN.test(staticMethodCall)) { + val expectedBooleanResult = expectedCallExpression.getAllTheSameExpectedBooleanConstants() ?: continue + val replacementMethod = if (expectedBooleanResult) mapping.replacementForTrue else mapping.replacementForFalse ?: return + registerMoveOutMethod(holder, expectedCallExpression, assertThatArgument, replacementMethod) { desc, method -> + MoveOutMethodCallExpressionQuickFix(desc, method) + } + } else if (mapping.expectNullNonNull != null) { + val expectedNullNonNullResult = expectedCallExpression.getExpectedNullNonNullResult() ?: continue + val replacementMethod = if (expectedNullNonNullResult xor mapping.expectNullNonNull) mapping.replacementForTrue else mapping.replacementForFalse ?: continue + registerMoveOutMethod(holder, expectedCallExpression, assertThatArgument, replacementMethod) { desc, method -> + MoveOutMethodCallExpressionQuickFix(desc, method, useNullNonNull = true) + } + } else if (mapping.expectedMatcher?.test(expectedCallExpression) == true) { + registerMoveOutMethod(holder, expectedCallExpression, assertThatArgument, mapping.replacementForTrue) { desc, method -> + MoveOutMethodCallExpressionQuickFix(desc, method, replaceOnlyThisMethod = mapping.expectedMatcher) + } + } + } + } + } + } + + private class Mapping( + val callMatcher: CallMatcher, + val replacementForTrue: String, + val replacementForFalse: String? = null, + val expectBoolean: Boolean = false, + val expectNullNonNull: Boolean? = null, + val expectedMatcher: CallMatcher? = null + ) +} \ No newline at end of file diff --git a/src/main/java/de/platon42/intellij/plugins/cajon/quickfixes/MoveOutMethodCallExpressionQuickFix.kt b/src/main/java/de/platon42/intellij/plugins/cajon/quickfixes/MoveOutMethodCallExpressionQuickFix.kt index c8d3fe5..52b1cf4 100644 --- a/src/main/java/de/platon42/intellij/plugins/cajon/quickfixes/MoveOutMethodCallExpressionQuickFix.kt +++ b/src/main/java/de/platon42/intellij/plugins/cajon/quickfixes/MoveOutMethodCallExpressionQuickFix.kt @@ -4,6 +4,7 @@ import com.intellij.codeInspection.ProblemDescriptor import com.intellij.openapi.project.Project import com.intellij.psi.JavaPsiFacade import com.intellij.psi.PsiMethodCallExpression +import com.siyeh.ig.callMatcher.CallMatcher import de.platon42.intellij.plugins.cajon.* class MoveOutMethodCallExpressionQuickFix( @@ -11,7 +12,8 @@ class MoveOutMethodCallExpressionQuickFix( private val replacementMethod: String, private val useNullNonNull: Boolean = false, private val noExpectedExpression: Boolean = false, - private val keepExpectedAsSecondArgument: Boolean = false + private val keepExpectedAsSecondArgument: Boolean = false, + private val replaceOnlyThisMethod: CallMatcher? = null ) : AbstractCommonQuickFix(description) { @@ -29,29 +31,46 @@ class MoveOutMethodCallExpressionQuickFix( val assertExpression = assertThatMethodCall.firstArg as? PsiMethodCallExpression ?: return val assertExpressionArg = if (noExpectedExpression) null else assertExpression.getArgOrNull(0)?.copy() - if (keepExpectedAsSecondArgument) { - assertExpressionArg ?: return - val secondArg = - if (useNullNonNull) JavaPsiFacade.getElementFactory(project).createExpressionFromText("null", null) else outmostCallExpression.getArgOrNull(0)?.copy() ?: return + when { + replaceOnlyThisMethod != null -> { + val methodsToFix = assertThatMethodCall.collectMethodCallsUpToStatement() + .filter(replaceOnlyThisMethod::test) + .toList() - assertExpression.replace(assertExpression.qualifierExpression) + assertExpression.replace(assertExpression.qualifierExpression) - val expectedExpression = createExpectedMethodCall(outmostCallExpression, replacementMethod, assertExpressionArg, secondArg) - expectedExpression.replaceQualifierFromMethodCall(outmostCallExpression) - outmostCallExpression.replace(expectedExpression) - } else { - val methodsToFix = assertThatMethodCall.collectMethodCallsUpToStatement() - .filter { (if (useNullNonNull) it.getExpectedNullNonNullResult() else it.getExpectedBooleanResult()) != null } - .toList() + methodsToFix + .forEach { + val expectedExpression = createExpectedMethodCall(it, replacementMethod, *it.argumentList.expressions) + expectedExpression.replaceQualifierFromMethodCall(it) + it.replace(expectedExpression) + } + } + keepExpectedAsSecondArgument -> { + assertExpressionArg ?: return + val secondArg = + if (useNullNonNull) JavaPsiFacade.getElementFactory(project).createExpressionFromText("null", null) else outmostCallExpression.getArgOrNull(0)?.copy() ?: return - assertExpression.replace(assertExpression.qualifierExpression) + assertExpression.replace(assertExpression.qualifierExpression) - methodsToFix - .forEach { - val expectedExpression = createExpectedMethodCall(it, replacementMethod, *listOfNotNull(assertExpressionArg).toTypedArray()) - expectedExpression.replaceQualifierFromMethodCall(it) - it.replace(expectedExpression) - } + val expectedExpression = createExpectedMethodCall(outmostCallExpression, replacementMethod, assertExpressionArg, secondArg) + expectedExpression.replaceQualifierFromMethodCall(outmostCallExpression) + outmostCallExpression.replace(expectedExpression) + } + else -> { + val methodsToFix = assertThatMethodCall.collectMethodCallsUpToStatement() + .filter { (if (useNullNonNull) it.getExpectedNullNonNullResult() else it.getExpectedBooleanResult()) != null } + .toList() + + assertExpression.replace(assertExpression.qualifierExpression) + + methodsToFix + .forEach { + val expectedExpression = createExpectedMethodCall(it, replacementMethod, *listOfNotNull(assertExpressionArg).toTypedArray()) + expectedExpression.replaceQualifierFromMethodCall(it) + it.replace(expectedExpression) + } + } } } } \ No newline at end of file diff --git a/src/main/resources/META-INF/plugin.xml b/src/main/resources/META-INF/plugin.xml index d29b894..f53d42a 100644 --- a/src/main/resources/META-INF/plugin.xml +++ b/src/main/resources/META-INF/plugin.xml @@ -45,6 +45,8 @@ implementationClass="de.platon42.intellij.plugins.cajon.inspections.AssertThatStringExpressionInspection"/>