JavaScript style guide and ESLint config.
This project contains my personal ESLint config. Every ESLint rule is documented in the base config file, regardless of whether or not it's enabled. Documentation consists of analysis, justification, and guidance. In some instances, this amounts to little more than paraphrasing of the official ESLint documentation. However, in many cases, there are fresh insights to be gained.
Prettier was added in v10 of this project even though the style it enforces is sometimes contrary to what was previously enforced by this project via ESLint rules. This change was made due to the enormous value Prettier adds in terms of consistency and accessibility across the JavaScript community.
It's unlikely that this ESLint config will perfectly suit your tastes. Many of the rules are subjective, some extremely so. That's OK, because this project isn't intended to serve as the standard for anyone's projects but my own. But that doesn't mean it can't be of any use to you. An enormous amount of time went into the research and deliberation of these rules, and it's possible that the resulting documentation can help you to define your own ESLint config, even if it ends up being contrary to mine.
My values have changed significantly over the years as I've progressed from creating solo projects at home, to writing software at work that must be understood, debugged, and modified by system administrators (instead of developers), to finally working on popular open source libraries that are both used and contributed to by developers of widely varying skill levels and ideologies.
Ranked by priority, I value:
- Security
- Correctness
- Debuggability
- Testability
- Maintainability
- Expressiveness
- Consistency
- Accessibility
- Elegance
- Performance
The only thing more important than correctness is security. The cost of exposing clients and/or the server to vulnerabilities is usually greater than the benefit provided by the code.
Some ESLint rules disallow practices that increase the risk of writing insecure code. But these rules are just a small part of creating secure software.
Obviously, the code needs to work.
Some ESLint rules disallow practices that increase the risk of writing buggy software. Other rules identify potential bugs in the code.
When things go wrong, we need to find out why. Quickly. There's more urgency involved with debugging than any other aspect of software development. Therefore, it's critical to optimize code in terms of debuggability. To make even a small sacrifice in debuggability, there must be tremendous gain elsewhere.
At a high level, debuggability is about the overall architect of the application. A debuggable application is modularized with clear separation of concerns between components.
At a low level, debuggability is about writing code with an intuitive logic flow, and avoiding practices that obfuscate stack traces when an error is thrown.
ESLint rules related to debuggability exist only at a low level.
There aren't many ESLint rules that impact testability, but it's still worth noting that writing testable code is a requirement for maintaining correctness. In particular, code must be modularized with clearly defined contracts. This facilitates unit testing and stubbing.
In order for code to be maintainable, it must be easy to refactor and build upon. A big part of maintainability exists at a high level in terms of the overall architecture. But there are low level practices too, such as avoiding duplication, and adhering to a style that produces clean diffs when changes are made.
ESLint rules related to maintainability exist only at a low level.
Good code doesn't create doubt in the minds of other developers, or our future selves. Instead, it expresses intent. And when intent can't be sufficiently expressed through code alone, then it's supplemented with comments.
Many ESLint rules work to eliminate ambiguity and doubt. However, there's a cost to expressiveness: verbosity. The extra code that often accompanies an expressive style can be frustrating, especially for those who place a lot of value in elegance. But the cost is worth the benefit.
Also known as the principle of least surprise, what this value boils down to is not defying the expectations of other developers. If something works a certain way in other applications, then it should work the same way in our applications. Furthermore, if something is handled a certain way in one part of our application, then it should be handled the same way in other parts.
Many ESLint rules disallow practices that create confusion, such as modifying builtin globals. Other rules enforce a consistent style throughout the code.
Every new thing that's added to a project, whether it be a language feature, framework, or library, reduces the accessibility of the code. Sometimes all that's gained in return is elegance or performance. This often isn't a good exchange.
The question isn't: "Does this new thing add any value?"
The question is: "Does this new thing add enough value?"
Sometimes the answer is yes. But often it's no.
Many ESLint rules disallow language features or quirks that add more complexity than value.
Elegance in code matters. It just doesn't matter as much as other values.
Many ESLint rules disallow language features or coding styles that make the code more concise but at too high of a cost in terms of other values.
Performance doesn't matter except when it does.
In most situations, a small gain in performance isn't worth whatever sacrifice that must be made to achieve it. But there are exceptions. Performance is the one value that can suddenly and drastically increase in priority, all the way from last to second, not replacing correctness but rather combining with it. That's because when there's a problem with performance, it's usually so severe that the software no longer fulfills its requirements, and is thus no longer correct. In such cases, the performance problem must be fixed at almost any cost.
In some projects, performance is so critical that it should be assumed forever intertwined with correctness. But for the vast majority of projects, in the vast majority of situations, no other value should be sacrificed in the name of performance.
Although there aren't any ESLint rules that directly address performance, some do have a minor performance impact.
My ESLint config is frequently updated as new versions of ESLint are released, as well as when I learn new things, or change my mind about existing rules.
npm install --save-dev eslint-config-tt
And if you haven't already:
npm install --save-dev eslint
Modify the "extends" directive in your ESLint config file to include "tt".
Example .eslintrc.yaml:
extends: tt
env:
node: true
There's also a set of modified rules available for Mocha / Chai / Sinon.
Example test/.eslintrc.yaml:
extends: tt/test.yaml
Example package.json:
{
"scripts": {
"lint": "eslint --fix ."
}
}
Now run npm run lint