One bite at a time - How to introduce new lint rules in a large codebase

christiankohler

Christian Kohler

Posted on November 26, 2019

One bite at a time - How to introduce new lint rules in a large codebase

Linters such as ESLint or TSLint can help make your code more readable and maintainable, and help you detect errors earlier. It's good to use linters from the beginning, but it's also a good idea to introduce linters into an existing code base.

This article focuses on the introduction of linters into existing code bases.

tl;dr;

  • Use autofix if possible
  • Extend lint config with a second config
  • Add new rules to the second config
  • Run linter with the second config with a precommit hook

The problem

Let's say the codebase is 1000 files large. You create a linter config, run the linter and you get like 1000000 errors. 🤯😱

Now what can you do?

Autofix

A lot of linting rule can be autofixed. For example the tslint rule

"no-var-keyword": true

can be autofixed. The autofixer replaces the var keyword with a let keyword.

Tip: All autofixable keywords in the list have the "Has Fixer" tag.

Manually fix

If you can't fix it automatically, you have to fix it manually. This can be a "Herculean task". So what often happens is that a rule is simply not used because it's too much work to fix all existing bugs.

The solution: The Boy Scout Rule

Leave your code better than you found it. ...

The boy scout approach to apply new rules is:

  • Fix existing errors when you touch existing code
  • Don't add new errors

Different Rules for CI/CD and changed files

We need two sets of rules. The main one and one which extends it and adds new rules.

Name Usage Rules
tslint.json CI/CD Rules which apply for all files
tslint.newrules.json precommit hook New rules which only apply for changed files

Example tslint.json

Used by the CI/CD pipeline.

{
  "defaultSeverity": "error",
  "rules": {
    "no-empty": true
  }
}

Example tslint.newrules.json

Used during the precommit hook.

{
  "defaultSeverity": "error",
  "extends": ["https://raw.githubusercontent.com/ChristianKohler/Homepage/master/content/posts/2019-11-25-lint-one-bite-at-a-time/tslint.json"],
  "rules": {
    "no-any": true
  }
}

Important: The tslint.newrules.json extends the main ruleset.

{
  "extends": ["https://raw.githubusercontent.com/ChristianKohler/Homepage/master/content/posts/2019-11-25-lint-one-bite-at-a-time/tslint.json"]
}

Enforce tslint.newrules.json with a precommit hook

This part is very easy nowadays thanks to the amazing libraries lint-staged and husky.

So just install lint-staged and then configure the precommit hook to run tslint or eslint with the the correct configuration.

npm install --save-dev lint-staged@beta
{
  "lint-staged": {
    "**/*.{ts}": ["tslint --project tsconfig.json -c tslint.newrules.json"]
  },
  "husky": {
    "hooks": {
      "pre-commit": "lint-staged"
    }
  }
}

Summary

It is easy and very little work to set up a "newrule" configuration and enforce the configuration with a pre-commit hook. Now your codebase should get better every day as people work on it. Without the upfront costs, you would have to fix all the bugs in a commit. That's how you eat an elephant. One bite at a time.

“How do you eat an elephant? One bite at a time.”

* I am strongly against eating elephants. It's a saying. Google it 😉

Hero photo by @keilahoetzel

💖 💪 🙅 🚩
christiankohler
Christian Kohler

Posted on November 26, 2019

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related