
io.gitlab.arturbosch.detekt.rules.style.MultilineLambdaItParameter.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.IT_LITERAL
import io.gitlab.arturbosch.detekt.rules.hasImplicitParameterReference
import io.gitlab.arturbosch.detekt.rules.implicitParameter
import org.jetbrains.kotlin.psi.KtLambdaExpression
import org.jetbrains.kotlin.resolve.BindingContext
/**
* Lambda expressions are very useful in a lot of cases and they often include very small chunks of
* code using only one parameter. In this cases Kotlin can supply the implicit `it` parameter
* to make code more concise. However, when you are dealing with lambdas that contain multiple statements,
* you might end up with a code that is hard to read if you don't specify a readable, descriptive parameter name
* explicitly.
*
*
* val digits = 1234.let {
* println(it)
* listOf(it)
* }
*
* val digits = 1234.let { it ->
* println(it)
* listOf(it)
* }
*
* val flat = listOf(listOf(1), listOf(2)).mapIndexed { index, it ->
* println(it)
* it + index
* }
*
*
*
* val digits = 1234.let { explicitParameterName ->
* println(explicitParameterName)
* listOf(explicitParameterName)
* }
*
* val lambda = { item: Int, that: String ->
* println(item)
* item.toString() + that
* }
*
* val digits = 1234.let { listOf(it) }
* val digits = 1234.let {
* listOf(it)
* }
* val digits = 1234.let { it -> listOf(it) }
* val digits = 1234.let { it ->
* listOf(it)
* }
* val digits = 1234.let { explicit -> listOf(explicit) }
* val digits = 1234.let { explicit ->
* listOf(explicit)
* }
*
*
*/
@RequiresTypeResolution
class MultilineLambdaItParameter(val config: Config) : Rule(config) {
override val issue = Issue(
javaClass.simpleName,
Severity.Style,
"Multiline lambdas should not use `it` as a parameter name",
Debt.FIVE_MINS
)
override fun visitLambdaExpression(lambdaExpression: KtLambdaExpression) {
super.visitLambdaExpression(lambdaExpression)
if (bindingContext == BindingContext.EMPTY) return
// If the lambda expression has <= 1 statements, skip check.
if (lambdaExpression.bodyExpression?.statements?.size ?: 0 <= 1) return
val parameterNames = lambdaExpression.valueParameters.map { it.name }
when {
// Explicit `it`
IT_LITERAL in parameterNames ->
report(
CodeSmell(
issue,
Entity.from(lambdaExpression),
"The parameter name in a multiline lambda should not be an explicit `it`. " +
"Consider giving your parameter a readable and descriptive name."
)
)
// Implicit `it`
parameterNames.isEmpty() -> {
val implicitParameter = lambdaExpression.implicitParameter(bindingContext)
if (implicitParameter != null &&
lambdaExpression.hasImplicitParameterReference(implicitParameter, bindingContext)
) {
report(
CodeSmell(
issue,
Entity.from(lambdaExpression),
"The implicit `it` should be used in a multiline lambda. " +
"Consider giving your parameter a readable and descriptive name."
)
)
}
}
else -> { }
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy