
io.gitlab.arturbosch.detekt.rules.style.ObjectLiteralToLambda.kt Maven / Gradle / Ivy
package io.gitlab.arturbosch.detekt.rules.style
import io.gitlab.arturbosch.detekt.api.CodeSmell
import io.gitlab.arturbosch.detekt.api.Config
import io.gitlab.arturbosch.detekt.api.Debt
import io.gitlab.arturbosch.detekt.api.Entity
import io.gitlab.arturbosch.detekt.api.Issue
import io.gitlab.arturbosch.detekt.api.Rule
import io.gitlab.arturbosch.detekt.api.Severity
import io.gitlab.arturbosch.detekt.api.internal.RequiresTypeResolution
import io.gitlab.arturbosch.detekt.rules.isOverride
import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.DeclarationDescriptor
import org.jetbrains.kotlin.psi.KtExpression
import org.jetbrains.kotlin.psi.KtNamedFunction
import org.jetbrains.kotlin.psi.KtObjectDeclaration
import org.jetbrains.kotlin.psi.KtObjectLiteralExpression
import org.jetbrains.kotlin.psi.KtThisExpression
import org.jetbrains.kotlin.psi.psiUtil.anyDescendantOfType
import org.jetbrains.kotlin.resolve.BindingContext
import org.jetbrains.kotlin.resolve.calls.callUtil.getResolvedCall
import org.jetbrains.kotlin.resolve.scopes.receivers.ImplicitClassReceiver
import org.jetbrains.kotlin.resolve.scopes.receivers.ReceiverValue
import org.jetbrains.kotlin.types.KotlinType
/**
* An anonymous object that does nothing other than the implementation of a single method
* can be used as a lambda.
*
* See https://kotlinlang.org/docs/java-interop.html#sam-conversions
* See https://kotlinlang.org/docs/fun-interfaces.html
*
*
* object : Foo {
* override fun bar() {
* }
* }
*
*
*
* Foo {
* }
*
*/
@RequiresTypeResolution
class ObjectLiteralToLambda(config: Config = Config.empty) : Rule(config) {
override val issue = Issue(
javaClass.simpleName,
Severity.Style,
"Report object literals that can be changed to lambdas.",
Debt.FIVE_MINS
)
private val KotlinType.couldBeSamInterface
get() = (constructor.declarationDescriptor as ClassDescriptor)
.isDefinitelyNotSamInterface
.not()
private fun KotlinType.singleSuperTypeOrNull(): KotlinType? =
constructor.supertypes.singleOrNull()
private fun KtObjectDeclaration.singleNamedMethodOrNull(): KtNamedFunction? =
declarations.singleOrNull() as? KtNamedFunction
private fun KtExpression.containsThisReference(descriptor: DeclarationDescriptor) =
anyDescendantOfType { thisReference ->
bindingContext[BindingContext.REFERENCE_TARGET, thisReference.instanceReference] == descriptor
}
private fun KtExpression.containsOwnMethodCall(descriptor: DeclarationDescriptor) =
anyDescendantOfType {
it.getResolvedCall(bindingContext)?.let { resolvedCall ->
resolvedCall.dispatchReceiver.isImplicitClassFor(descriptor) ||
resolvedCall.extensionReceiver.isImplicitClassFor(descriptor)
} == true
}
private fun ReceiverValue?.isImplicitClassFor(descriptor: DeclarationDescriptor) =
this is ImplicitClassReceiver && classDescriptor == descriptor
private fun KtExpression.containsMethodOf(declaration: KtObjectDeclaration): Boolean {
val objectDescriptor = bindingContext[BindingContext.DECLARATION_TO_DESCRIPTOR, declaration]
?: return false
return containsThisReference(objectDescriptor) ||
containsOwnMethodCall(objectDescriptor)
}
private fun KtObjectDeclaration.hasConvertibleMethod(): Boolean {
val singleNamedMethod = singleNamedMethodOrNull()
val functionBody = singleNamedMethod?.bodyExpression ?: return false
return singleNamedMethod.isOverride() &&
!functionBody.containsMethodOf(this)
}
override fun visitObjectLiteralExpression(expression: KtObjectLiteralExpression) {
super.visitObjectLiteralExpression(expression)
if (bindingContext == BindingContext.EMPTY) return
val declaration = expression.objectDeclaration
if (
declaration.name == null &&
bindingContext.getType(expression)
?.singleSuperTypeOrNull()?.couldBeSamInterface == true &&
declaration.hasConvertibleMethod()
) {
report(CodeSmell(issue, Entity.from(expression), issue.description))
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy