Skip to content

Commit

Permalink
Added ReplaceAssertBooleanWithAssertEqualityInspection
Browse files Browse the repository at this point in the history
  • Loading branch information
tommykw committed Aug 4, 2018
1 parent e4bf20a commit 65f3f3d
Show file tree
Hide file tree
Showing 24 changed files with 324 additions and 0 deletions.
@@ -0,0 +1,6 @@
<html>
<body>
This inspection reports assert boolean function calls replaceable with assert equality function.
Example: <b>assertTrue(a == b)</b> can be replaced by <b>assertEquals(a, b)</b>.
</body>
</html>
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml
Expand Up @@ -2967,6 +2967,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml.173
Expand Up @@ -2966,6 +2966,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml.181
Expand Up @@ -2966,6 +2966,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml.183
Expand Up @@ -2967,6 +2967,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml.as31
Expand Up @@ -2966,6 +2966,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
9 changes: 9 additions & 0 deletions idea/src/META-INF/plugin.xml.as32
Expand Up @@ -2966,6 +2966,15 @@ The Kotlin plugin provides language support in IntelliJ IDEA and Android Studio.
language="kotlin"
/>

<localInspection implementationClass="org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection"
displayName="Replace assert boolean with assert equality"
groupPath="Kotlin"
groupName="Style issues"
enabledByDefault="true"
level="INFORMATION"
language="kotlin"
/>

<referenceImporter implementation="org.jetbrains.kotlin.idea.quickfix.KotlinReferenceImporter"/>

<fileType.fileViewProviderFactory filetype="KJSM" implementationClass="com.intellij.psi.ClassFileViewProviderFactory"/>
Expand Down
@@ -0,0 +1,84 @@
/*
* Copyright 2010-2018 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license
* that can be found in the license/LICENSE.txt file.
*/

package org.jetbrains.kotlin.idea.inspections

import com.intellij.openapi.editor.Editor
import com.intellij.openapi.project.Project
import com.intellij.psi.PsiElement
import org.jetbrains.kotlin.idea.intentions.getCallableDescriptor
import org.jetbrains.kotlin.lexer.KtTokens
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.psi.*
import org.jetbrains.kotlin.resolve.descriptorUtil.fqNameSafe

class ReplaceAssertBooleanWithAssertEqualityInspection : AbstractApplicabilityBasedInspection<KtCallExpression>(KtCallExpression::class.java) {

override fun inspectionText(element: KtCallExpression) = "Replace assert boolean with assert equality"

override val defaultFixText = "Replace assert boolean with assert equality"

override fun fixText(element: KtCallExpression): String {
val assertion = element.replaceableAssertion() ?: return defaultFixText
return "Replace with '$assertion'"
}

override fun isApplicable(element: KtCallExpression): Boolean {
return (element.replaceableAssertion() != null)
}

override fun applyTo(element: PsiElement, project: Project, editor: Editor?) {
val expression = element as? KtCallExpression ?: return
val condition = expression.valueArguments.first().getArgumentExpression() as? KtBinaryExpression ?: return
val left = condition.left ?: return
val right = condition.right ?: return
val assertion = expression.replaceableAssertion() ?: return
val factory = KtPsiFactory(project)

if (expression.valueArguments.size == 1) {
expression.replace(factory.createExpressionByPattern("$assertion($0, $1)", left, right))
} else if (expression.valueArguments.size == 2) {
val message = expression.valueArguments[1].getArgumentExpression() ?: return
expression.replace(factory.createExpressionByPattern("$assertion($0, $1, $2)", left, right, message))
}
}

private fun KtCallExpression.replaceableAssertion(): String? {
var result: String? = null
val referencedName = (calleeExpression as? KtNameReferenceExpression)?.getReferencedName() ?: return null
if (!assertions.contains(referencedName)) {
return result
}

if (getCallableDescriptor()?.containingDeclaration?.fqNameSafe != FqName("kotlin.test")) {
return result
}

if (valueArguments.size != 1 && valueArguments.size != 2) return null
val binaryExpression = valueArguments.first().getArgumentExpression() as? KtBinaryExpression ?: return null
val operationToken = binaryExpression.operationToken

assertionMap.entries.forEach {
val (assertion, token) = it.key
if (referencedName == assertion && operationToken == token) {
result = it.value
return@forEach
}
}

return result
}

companion object {
private val assertions = setOf("assertTrue", "assertFalse")

private val assertionMap = mapOf(
Pair("assertTrue", KtTokens.EQEQ) to "assertEquals",
Pair("assertTrue", KtTokens.EQEQEQ) to "assertSame",
Pair("assertFalse", KtTokens.EQEQ) to "assertNotEquals",
Pair("assertFalse", KtTokens.EQEQEQ) to "assertNotSame"
)
}
}
@@ -0,0 +1 @@
org.jetbrains.kotlin.idea.inspections.ReplaceAssertBooleanWithAssertEqualityInspection
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST
// PROBLEM: none

import kotlin.test.*

fun foo() {
val isA = false
assertFalse<caret>(isA)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertFalse<caret>(a == b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertNotEquals(a, b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertFalse<caret>(a === b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertNotSame(a, b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertFalse(<caret>a == b, "message")
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "b"
assertNotEquals(a, b, "message")
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST
// PROBLEM: none

import kotlin.test.*

fun foo() {
val isA = true
assertTrue<caret>(isA)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertTrue<caret>(a == b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertEquals(a, b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertTrue(<caret>a === b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertSame(a, b)
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertTrue(<caret>a == b, "message")
}
@@ -0,0 +1,9 @@
// RUNTIME_WITH_KOTLIN_TEST

import kotlin.test.*

fun foo() {
val a = "a"
val b = "a"
assertEquals(a, b, "message")
}

0 comments on commit 65f3f3d

Please sign in to comment.