Published on

|

10 min

Building Custom Lint Rules for Kotlin: Mastering Code Quality in Android Development

Ashutosh Makwana
Ashutosh Makwana

As an Android developer with over 5 years of experience, I've seen firsthand how crucial code quality is for maintaining large-scale projects. One tool that's been indispensable in my journey is Android Lint. But sometimes, the built-in rules just aren't enough. That's where custom Lint rules come in. In this guide, I'll walk you through creating powerful custom Lint rules for Kotlin that will elevate your team's coding standards and streamline your development process.
Cover Image for Building Custom Lint Rules for Kotlin: Mastering Code Quality in Android Development

Why Custom Lint Rules Matter

Before we dive in, let's talk about why custom Lint rules are so important. In my experience, they've been game-changers for:

  1. Enforcing team-wide coding standards

  2. Catching project-specific issues early

  3. Improving code readability and maintainability

  4. Reducing the time spent on code reviews

Now, let's get our hands dirty and start building some custom Lint rules!

Setting Up Your Custom Lint Module

First things first, we need to set up a dedicated module for our custom Lint rules. Here's how I usually do it:

  1. In Android Studio, go to File > New > New Module

  2. Choose "Java or Kotlin Library"

  3. Name it something like "lint-rules"

Once you've got your module, add these dependencies to its build.gradle file:

groovy

dependencies {
    compileOnly "com.android.tools.lint:lint-api:30.0.0"
    compileOnly "com.android.tools.lint:lint-checks:30.0.0"
}

Creating Your First Custom Lint Rule

Let's start with a simple rule that checks for proper naming conventions. We'll create a detector that ensures class names are in UpperCamelCase.

kotlin

class NamingConventionDetector : Detector(), SourceCodeScanner {
    companion object {
        val ISSUE = Issue.create(
            id = "ImproperClassName",
            briefDescription = "Class names should be UpperCamelCase",
            explanation = "Following the Android naming convention, class names should start with an uppercase letter and use camel case.",
            category = Category.CORRECTNESS,
            priority = 6,
            severity = Severity.WARNING,
            implementation = Implementation(NamingConventionDetector::class.java, Scope.JAVA_FILE_SCOPE)
        )
    }

    override fun appliesTo(folderType: ResourceFolderType) = true

    override fun visitClass(context: JavaContext, declaration: UClass) {
        if (!declaration.name!!.matches("[A-Z][a-zA-Z0-9]*".toRegex())) {
            context.report(
                ISSUE,
                context.getNameLocation(declaration),
                "Class name should be in UpperCamelCase"
            )
        }
    }
}

This detector checks each class declaration and reports an issue if the class name doesn't follow the UpperCamelCase convention.

Registering Your Custom Lint Rules

To make Lint aware of your custom rules, you need to create an IssueRegistry:

kotlin

class CustomIssueRegistry : IssueRegistry() {
    override val issues: List<Issue>
        get() = listOf(NamingConventionDetector.ISSUE)
}

Don't forget to declare this registry in your build.gradle:

groovy

jar {
    manifest {
        attributes("Lint-Registry-v2": "com.yourpackage.CustomIssueRegistry")
    }
}

Advanced Lint Techniques: AST Analysis

As you get more comfortable with basic rules, you might want to dive into more advanced techniques. One powerful tool in your Lint arsenal is Abstract Syntax Tree (AST) analysis. This allows you to examine the structure of the code at a deeper level.

Here's an example of a more advanced rule that checks for empty catch blocks:

kotlin

class EmptyCatchDetector : Detector(), SourceCodeScanner {
    companion object {
        val ISSUE = Issue.create(
 
           id = "EmptyCatchBlock",
            briefDescription = "Empty catch block",
            explanation = "Empty catch blocks can swallow exceptions and hide errors. Consider logging or rethrowing the exception.",
            category = Category.CORRECTNESS,
            priority = 5,
            severity = Severity.WARNING,
            implementation = Implementation(EmptyCatchDetector::class.java, Scope.JAVA_FILE_SCOPE)
        )
    }

    override fun getApplicableUastTypes() = listOf(UTryExpression::class.java)

    override fun createUastHandler(context: JavaContext) = object : UElementHandler() {
        override fun visitTryExpression(node: UTryExpression) {
            node.catchClauses.forEach { catchClause ->
                if (catchClause.body?.expressions?.isEmpty() == true) {
                    context.report(
                        ISSUE,
                        context.getLocation(catchClause.body),
                        "This catch block is empty"
                    )
                }
            }
        }
    }
}

This detector uses the AST to examine try-catch blocks and reports an issue if it finds an empty catch block.

Implementing Project-Specific Rules

One of the biggest advantages of custom Lint rules is the ability to enforce project-specific standards. For example, let's say your team has decided to avoid using a certain deprecated API. You can create a rule for that:

kotlin

class DeprecatedApiDetector : Detector(), SourceCodeScanner {
    companion object {
        val ISSUE = Issue.create(
            id = "DeprecatedApiUsage",
            briefDescription = "Usage of deprecated API",
            explanation = "This API has been deprecated and should not be used in new code.",
            category = Category.CORRECTNESS,
            priority = 7,
            severity = Severity.ERROR,
            implementation = Implementation(DeprecatedApiDetector::class.java, Scope.JAVA_FILE_SCOPE)
        )
    }

    override fun getApplicableMethodNames() = listOf("deprecatedMethod")

    override fun visitMethodCall(context: JavaContext, node: UCallExpression, method: PsiMethod) {
        context.report(
            ISSUE,
            context.getLocation(node),
            "Use of deprecated API 'deprecatedMethod'. Use 'newMethod' instead."
        )
    }
}

This rule will flag any usage of a method named deprecatedMethod, suggesting to use newMethod instead.

Testing Your Custom Lint Rules

Testing is crucial to ensure your Lint rules work as expected. Here's how you can set up a test for our NamingConventionDetector:

kotlin

class NamingConventionDetectorTest : LintDetectorTest() {
    override fun getDetector() = NamingConventionDetector()
    override fun getIssues() = listOf(NamingConventionDetector.ISSUE)

    fun testImproperClassName() {
        lint().files(
            kotlin("""
                class invalidClassName {
                    
// ...
                }
            """.trimIndent())
        ).run()
         .expect("""
             src/invalidClassName.kt:1: Warning: Class name should be in UpperCamelCase [ImproperClassName]
             class invalidClassName {
                   ~~~~~~~~~~~~~~~~
             0 errors, 1 warnings
         """.trimIndent())
    }
}

This test ensures that our NamingConventionDetector correctly flags classes with improper names.

Integrating Custom Lint Rules into Your Workflow

To get the most out of your custom Lint rules, integrate them into your build process and CI/CD pipeline. Add this to your app's build.gradle:

groovy

android {
    lintOptions {
        checkDependencies true
        lintConfig file("lint.xml")
        htmlOutput file("lint-report.html")
        abortOnError true
    }
}

This configuration will run your custom Lint checks as part of the build process and generate an HTML report.

Best Practices for Custom Lint Rules

After years of working with custom Lint rules, here are some best practices I've learned:

  1. Start small: Begin with a few crucial rules and gradually expand.

  2. Provide clear explanations: Make sure your error messages are informative and actionable.

  3. Use appropriate severity levels: Not every issue needs to be an error.

  4. Keep rules up-to-date: Regularly review and update your rules as your project evolves.

  5. Document your rules: Maintain a wiki or README explaining each custom rule.

Balancing Code Quality and Productivity

While strict coding standards are important, it's crucial to strike a balance with developer productivity. Here are some tips:

  1. Prioritize rules that catch actual bugs over stylistic preferences.

  2. Provide auto-fix options where possible to save developers time.

  3. Regularly review and refine your rules based on team feedback.

  4. Use suppression annotations judiciously for false positives or exceptional cases.

Conclusion

Custom Lint rules have been a game-changer in my Android development career. They've helped me and my teams maintain high code quality, catch bugs early, and enforce consistent standards across large projects. While there's a learning curve, the long-term benefits in code maintainability and reduced technical debt are well worth the effort.

I hope this guide helps you get started with creating your own custom Lint rules. Remember, the goal is to improve your codebase and make your team more efficient, not to create unnecessary obstacles. Happy coding!.