A linter is a static code analysis tool that flags programming errors, bugs, stylistic errors, and suspicious constructs. Linters examine your code without actually executing it, helping you identify potential problems before runtime.
Think of linters as automated code reviewers that enforce consistent coding styles and catch common mistakes across your codebase. They work by parsing your code into an Abstract Syntax Tree (AST) and running various rules against that structure.
Linters bring tremendous value to development workflows for many reasons:
Linters identify issues as you write code, before you even save the file. This "shift-left" approach to debugging means:
// Without linting, this error might only be caught at runtime
function calculateTotal(prices) {
let total = 0;
// Typo in variable name would cause a runtime error
for (const price of prces) {
total += price;
}
return total;
}
// A linter would immediately flag the undefined variable 'prces'
When working on a team, maintaining a consistent code style is crucial for readability and maintainability.
// Inconsistent style example
function badlyFormattedFunction( name ) {
if(name){
return "Hello, "+name+"!";
}else{
return"Hello, stranger!";
}
}
// After linting - consistent style
function wellFormattedFunction(name) {
if (name) {
return `Hello, ${name}!`;
} else {
return 'Hello, stranger!';
}
}
Linters help developers follow best practices and avoid antipatterns:
// Problematic code with potential memory leak
function createButtons() {
for (var i = 0; i < 10; i++) {
var button = document.createElement('button');
button.innerText = 'Button ' + i;
// This creates closures over 'i' that may cause unexpected behavior
button.addEventListener('click', function() {
console.log('Clicked button ' + i);
});
document.body.appendChild(button);
}
}
// Linter would suggest using 'let' instead of 'var' and potentially
// other improvements to the event handling pattern
For new developers, linters provide real-time feedback that teaches proper syntax, style, and patterns:
When linters enforce style and catch basic issues, code reviewers can focus on architecture, logic, and design decisions instead of formatting and syntax issues.
A few popular linters are:
Many languages have their own linters, which we will not cover in this guide.
In this guide, we'll focus on ESLint.
ESLint is the most widely used JavaScript linter. It's highly configurable and extensible with plugins for various frameworks and libraries.
Key features:
Open your terminal in your project directory and run:
# Using npm
npm install eslint --save-dev
# Using yarn
yarn add eslint --dev
Run the ESLint initialization command:
# Using npx (comes with npm)
npx eslint --init
# Or if you installed ESLint globally
eslint --init
This will launch an interactive setup that asks several questions:
After answering, ESLint will create a configuration file (.eslintrc.js
, .eslintrc.json
, or .eslintrc.yml
) in your project.
To enable automatic fixing of issues when you save a file:
Or add this to your settings.json
:
{
"editor.codeActionsOnSave": {
"source.fixAll.eslint": true
}
}
First, when configuring ESLint, you can extend a popular style guide like Airbnb or Standard. This will automatically include many of the rules for you.
// .eslintrc.js
module.exports = {
extends: 'airbnb',
};
However, you may want to configure additional rules or override the defaults. Here are some popular rules you might want to configure:
// .eslintrc.js
module.exports = {
// Extend the Airbnb style guide
extends: 'airbnb',
// Any overrides go in the rules section
"rules": {
// Disallow unused variables
"no-unused-vars": "error",
// Require === instead of ==
"eqeqeq": "error",
// Disallow console.log statements
"no-console": "warn",
// Require default case in switch statements
"default-case": "warn"
}
};
// .eslintrc.js
module.exports = {
extends: 'airbnb',
"rules": {
// Enforce consistent indentation (2 spaces)
"indent": ["error", 2],
// Require semicolons
"semi": ["error", "always"],
// Use single quotes for strings
"quotes": ["error", "single"],
// Require space before blocks
"space-before-blocks": "error"
}
};