![JAR search and dependency download from the Maven repository](/logo.png)
io.gitlab.arturbosch.detekt.rules.style.UtilityClassWithPublicConstructor.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of detekt-rules-style Show documentation
Show all versions of detekt-rules-style Show documentation
Static code analysis for Kotlin
The newest version!
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.ActiveByDefault
import io.gitlab.arturbosch.detekt.rules.isOpen
import org.jetbrains.kotlin.psi.KtClass
import org.jetbrains.kotlin.psi.KtClassInitializer
import org.jetbrains.kotlin.psi.KtDeclaration
import org.jetbrains.kotlin.psi.KtObjectDeclaration
import org.jetbrains.kotlin.psi.KtSecondaryConstructor
import org.jetbrains.kotlin.psi.psiUtil.isPublic
/**
* A class which only contains utility variables and functions with no concrete implementation can be refactored
* into an `object` or a class with a non-public constructor.
* Furthermore, this rule reports utility classes which are not final.
*
*
* class UtilityClassViolation {
*
* // public constructor here
* constructor() {
* // ...
* }
*
* companion object {
* val i = 0
* }
* }
*
* open class UtilityClassViolation private constructor() {
*
* // ...
* }
*
*
*
* class UtilityClass {
*
* private constructor() {
* // ...
* }
*
* companion object {
* val i = 0
* }
* }
* object UtilityClass {
*
* val i = 0
* }
*
*/
@ActiveByDefault(since = "1.2.0")
class UtilityClassWithPublicConstructor(config: Config = Config.empty) : Rule(config) {
override val issue: Issue = Issue(
javaClass.simpleName,
Severity.Style,
"The class declaration is unnecessary because it only contains utility functions. " +
"An object declaration should be used instead.",
Debt.FIVE_MINS
)
override fun visitClass(klass: KtClass) {
if (canBeCheckedForUtilityClass(klass)) {
val utilityClassConstructor = UtilityClassConstructor(klass)
val declarations = klass.body?.declarations
if (hasOnlyUtilityClassMembers(declarations)) {
if (utilityClassConstructor.hasPublicConstructorWithoutParameters()) {
report(
CodeSmell(
issue,
Entity.from(klass),
"The class ${klass.nameAsSafeName} only contains" +
" utility functions. Consider defining it as an object."
)
)
} else if (klass.isOpen() && utilityClassConstructor.hasNonPublicConstructorWithoutParameters()) {
report(
CodeSmell(
issue,
Entity.from(klass),
"The utility class ${klass.nameAsSafeName} should be final."
)
)
}
}
}
super.visitClass(klass)
}
private fun canBeCheckedForUtilityClass(klass: KtClass): Boolean {
return !klass.isInterface() &&
!klass.superTypeListEntries.any() &&
!klass.isAnnotation() &&
!klass.isSealed()
}
private fun hasOnlyUtilityClassMembers(declarations: List?): Boolean {
if (declarations.isNullOrEmpty()) {
return false
}
var containsCompanionObject = false
var isUtilityClassCandidate = true
declarations.forEach {
if (isCompanionObject(it)) {
containsCompanionObject = true
} else if (it !is KtSecondaryConstructor && it !is KtClassInitializer) {
isUtilityClassCandidate = false
}
}
return containsCompanionObject && isUtilityClassCandidate
}
private fun isCompanionObject(declaration: KtDeclaration) =
(declaration as? KtObjectDeclaration)?.isCompanion() == true
internal class UtilityClassConstructor(private val klass: KtClass) {
internal fun hasPublicConstructorWithoutParameters() = hasConstructorWithoutParameters(true)
internal fun hasNonPublicConstructorWithoutParameters() = hasConstructorWithoutParameters(false)
private fun hasConstructorWithoutParameters(publicModifier: Boolean): Boolean {
val primaryConstructor = klass.primaryConstructor
if (primaryConstructor != null) {
return primaryConstructor.isPublic == publicModifier && primaryConstructor.valueParameters.isEmpty()
}
val secondaryConstructors = klass.secondaryConstructors
return secondaryConstructors.isEmpty() ||
secondaryConstructors.any { it.isPublic == publicModifier && it.valueParameters.isEmpty() }
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy