{"pageProps":{"readTime":{"text":"5 min read","minutes":4.73,"time":283800,"words":946},"source":{"compiledSource":"\"use strict\";\n\nfunction _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }\n\nfunction _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }\n\nfunction _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }\n\n/* @jsx mdx */\nvar layoutProps = {};\nvar MDXLayout = \"wrapper\";\n\nfunction MDXContent(_ref) {\n var components = _ref.components,\n props = _objectWithoutProperties(_ref, [\"components\"]);\n\n return mdx(MDXLayout, _extends({}, layoutProps, props, {\n components: components,\n mdxType: \"MDXLayout\"\n }), mdx(\"p\", null, \"If you\\u2019ve spent any amount of time configuring ESLint \\u2014 the most widely used linting tool for JavaScript \\u2014 you\\u2019ll know you can easily get lost in it.\"), mdx(\"p\", null, \"But there are also formatters like Prettier, which are excellent but often fight with linters.\"), mdx(\"p\", null, \"I strongly believe that a solid linting setup con save you tons of time and catch many bugs much sooner. The first time you save a file and it auto-formats like this, you'll see how powerful it is:\"), mdx(\"p\", null, mdx(\"img\", _extends({\n parentName: \"p\"\n }, {\n \"src\": \"/images/autofix-example.gif\",\n \"alt\": \"GIF showing linting / formatting tools fixing code automatically\"\n }))), mdx(\"p\", null, \"After much trial and error, I\\u2019ve put together and setup I use for every project and published it for easier use.\"), mdx(\"h3\", null, \"1. Set Up Your Project\"), mdx(\"p\", null, \"Make a new directory for your linting config and run either \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"yarn init -y\"), \" or \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"npm init -y\"), \" to create a \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"package.json\"), \".\"), mdx(\"p\", null, \"When publishing an ESLint config, all we need to export is an ESLint config object.\"), mdx(\"p\", null, \"So let's make an \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"index.js\"), \" file with the following content:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"module.exports = {\\n\\n}\\n\")), mdx(\"p\", null, \"And then install ES Lint:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-bash\"\n }), \"yarn add eslint\\n# or\\nnpm install --save eslint\\n\")), mdx(\"h3\", null, \"2. Find a setup to use as a base\"), mdx(\"p\", null, \"Instead of hand-picking every linter rule, I would strongly suggest finding an existing config and \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://eslint.org/docs/user-guide/configuring#extending-configuration-files\"\n }), \"extending it.\")), mdx(\"p\", null, \"I would strongly recommend using React's official \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://github.com/facebook/create-react-app/tree/master/packages/eslint-config-react-app#usage-outside-of-create-react-app\"\n }), \"eslint-config-react-app\"), \", but if you want to find more options you can check out \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://github.com/dustinspecker/awesome-eslint\"\n }), \"this repo.\"), \".\"), mdx(\"p\", null, \"For many years, I was a strong advocate for the AirBNB linting config, but recently I realized I'd been turning off many of their opinionated rules.\"), mdx(\"p\", null, \"I finally removed it after reading \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://overreacted.io/writing-resilient-components/#marie-kondo-your-lint-config\"\n }), \"Dan Abramov\\u2019s\"), \" advice on the topic:\"), mdx(\"blockquote\", null, mdx(\"p\", {\n parentName: \"blockquote\"\n }, \"Don\\u2019t assume that whatever you or something somebody else added to your lint config a year ago is a \\u201Cbest practice\\u201D. Question it and look for answers. Don\\u2019t let anyone tell you you\\u2019re not smart enough to pick your lint rules.\")), mdx(\"p\", null, \"If you do prefer more rules, then more power to you! But if you find they're getting in your way and not helping catch bugs, it might be time to part ways.\"), mdx(\"p\", null, \"In case you do go with a different config (and you use React), make sure to include this \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://github.com/evcohen/eslint-plugin-jsx-a11y\"\n }), \"jsx accessibility\"), \" plugin. Even if your primary focus isn't accessibility, I believe every project should have this by default even if it the rules seem frustrating at first.\"), mdx(\"p\", null, \"Once you have your base, update your \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"index.js\"), \":\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"module.exports = {\\n extends: ['react-app'] // or whatever you chose as your base config\\n}\\n\")), mdx(\"h3\", null, \"3. Add Your Formatter\"), mdx(\"p\", null, \"Now that we have our linter set up to catch code-related bugs, we're going to use \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://prettier.io/\"\n }), \"Prettier\"), \" to consistently format our code.\"), mdx(\"p\", null, \"First add these packages:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-bash\"\n }), \"yarn add prettier eslint-config-prettier eslint-plugin-prettier\\n# or\\nnpm install --save prettier eslint-config-prettier eslint-plugin-prettier\\n\")), mdx(\"p\", null, \"Here's what we've added:\"), mdx(\"ul\", null, mdx(\"li\", {\n parentName: \"ul\"\n }, mdx(\"inlineCode\", {\n parentName: \"li\"\n }, \"prettier\"), \" - Used to run the formatting\"), mdx(\"li\", {\n parentName: \"ul\"\n }, mdx(\"inlineCode\", {\n parentName: \"li\"\n }, \"eslint-config-prettier\"), \" - Turns off ESLint formatting rules that would conflict with Prettier\"), mdx(\"li\", {\n parentName: \"ul\"\n }, mdx(\"inlineCode\", {\n parentName: \"li\"\n }, \"eslint-plugin-prettier\"), \" - Allows your ESLint config to run prettier\")), mdx(\"p\", null, \"While the setup is a bit cumbersome, this will allow you to \", mdx(\"em\", {\n parentName: \"p\"\n }, \"just\"), \" use ESLint to lint and format your code and keep you from having to use Prettier separately.\"), mdx(\"p\", null, \"Now, update your \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"index.js\"), \" again:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"module.exports = {\\n extends: [\\n 'react-app', // or whatever you chose as your base config\\n 'plugin:prettier/recommended', // this will stop Prettier and ESLint from fighting over fixes\\n 'prettier/react', // optionally include this if you use React\\n ],\\n 'plugins': ['prettier'],\\n 'rules': {\\n // make Prettier return errors\\n 'prettier/prettier': [\\n 'error',\\n {\\n // Optional Prettier config changes\\n trailingComma: 'es5',\\n singleQuote: true,\\n printWidth: 80,\\n },\\n ],\\n }\\n}\\n\")), mdx(\"h3\", null, \"4. Publishing\"), mdx(\"p\", null, \"To publish this config, you'll need to update a few things in your \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"package.json\"), \":\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"{\\n // must start with \\\"eslint-config\\\" for other people to include it\\n \\\"name\\\": \\\"eslint-config-my-cool-config\\\",\\n // this must be increased every time you publish\\n \\\"version\\\": \\\"1.0.0\\\",\\n // this tells node where to find the main module when it's installed\\n \\\"main\\\": \\\"index.js\\\",\\n}\\n\")), mdx(\"p\", null, \"Then, make sure you have an account with \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://www.npmjs.com\"\n }), \"NPM\"), \" and follow their guide to \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://docs.npmjs.com/creating-and-publishing-unscoped-public-packages#publishing-unscoped-public-packages\"\n }), \"publish a package\"), \".\"), mdx(\"p\", null, \"Once it's successful, you should be able to go to any Javascript project using NPM and run:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-bash\"\n }), \"yarn add eslint-config-my-cool-config\\n# or\\nnpm install --save eslint-config-my-cool-config\\n\")), mdx(\"p\", null, \"And then extend your own config by making a \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \".eslintrc\"), \" file in the root of your project:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"{\\n // don't include the `eslint-config-` part when extending\\n \\\"extends\\\": [\\\"my-cool-config\\\"]\\n}\\n\")), mdx(\"h3\", null, \"5. Super Charge Your Text Editor\"), mdx(\"p\", null, \"If you aren't sold on the benefits of all this setup yet, this is where you'll start to see how helpful this is.\"), mdx(\"p\", null, \"Your text editor should have an official \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://eslint.org/docs/user-guide/integrations#editors\"\n }), \"ESLint integration\"), \", but for this example we're going to set it up in VS Code.\"), mdx(\"p\", null, \"Install the \", mdx(\"a\", _extends({\n parentName: \"p\"\n }, {\n \"href\": \"https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint\"\n }), \"VSCode ESLint extension\"), \" and then open the command pallette (\", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"cmd / ctrl + shift + p\"), \") and type \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"settings json\"), \" and select \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \"Open Settings (JSON)\"), \".\"), mdx(\"p\", null, \"At the end of the settings JSON, add this:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"\\\"editor.codeActionsOnSave\\\": {\\n \\\"source.fixAll.eslint\\\": true\\n},\\n\")), mdx(\"p\", null, \"Now when you save a file it should automatically run eslint fixes, which also include our Prettier formatting and save you hours of headache in the future.\"), mdx(\"p\", null, \"Finally, don't forget that if you end up changing your mind about any rules, you can always override them in your \", mdx(\"inlineCode\", {\n parentName: \"p\"\n }, \".eslintrc\"), \" project's config file:\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"{\\n // don't include the `eslint-config-` part when extending\\n \\\"extends\\\": [\\\"my-cool-config\\\"],\\n \\\"rules\\\": {\\n // 0 turns the rule off, 1 emits a console warning and 2 throws an error\\n \\\"overly-opinionated-rule\\\": 0\\n }\\n}\\n\")), mdx(\"p\", null, \"Or change them in your custom config package (just remember you'll have to publish it afterward):\"), mdx(\"pre\", null, mdx(\"code\", _extends({\n parentName: \"pre\"\n }, {\n \"className\": \"language-js\"\n }), \"{\\n \\\"extends\\\": [\\\"base-config\\\"],\\n \\\"rules\\\": {\\n \\\"overly-opinionated-rule\\\": 0\\n }\\n}\\n\")));\n}\n\n;\nMDXContent.isMDXComponent = true;","renderedOutput":"

If you’ve spent any amount of time configuring ESLint — the most widely used linting tool for JavaScript — you’ll know you can easily get lost in it.

But there are also formatters like Prettier, which are excellent but often fight with linters.

I strongly believe that a solid linting setup con save you tons of time and catch many bugs much sooner. The first time you save a file and it auto-formats like this, you'll see how powerful it is:

\"GIF

After much trial and error, I’ve put together and setup I use for every project and published it for easier use.

1. Set Up Your Project

Make a new directory for your linting config and run either yarn init -y or npm init -y to create a package.json.

When publishing an ESLint config, all we need to export is an ESLint config object.

So let's make an index.js file with the following content:

1module.exports = {
2
3}

And then install ES Lint:

1yarn add eslint
2# or
3npm install --save eslint

2. Find a setup to use as a base

Instead of hand-picking every linter rule, I would strongly suggest finding an existing config and extending it.

I would strongly recommend using React's official eslint-config-react-app, but if you want to find more options you can check out this repo..

For many years, I was a strong advocate for the AirBNB linting config, but recently I realized I'd been turning off many of their opinionated rules.

I finally removed it after reading Dan Abramov’s advice on the topic:

Don’t assume that whatever you or something somebody else added to your lint config a year ago is a “best practice”. Question it and look for answers. Don’t let anyone tell you you’re not smart enough to pick your lint rules.

If you do prefer more rules, then more power to you! But if you find they're getting in your way and not helping catch bugs, it might be time to part ways.

In case you do go with a different config (and you use React), make sure to include this jsx accessibility plugin. Even if your primary focus isn't accessibility, I believe every project should have this by default even if it the rules seem frustrating at first.

Once you have your base, update your index.js:

1module.exports = {
2 extends: ['react-app'] // or whatever you chose as your base config
3}

3. Add Your Formatter

Now that we have our linter set up to catch code-related bugs, we're going to use Prettier to consistently format our code.

First add these packages:

1yarn add prettier eslint-config-prettier eslint-plugin-prettier
2# or
3npm install --save prettier eslint-config-prettier eslint-plugin-prettier

Here's what we've added:

While the setup is a bit cumbersome, this will allow you to just use ESLint to lint and format your code and keep you from having to use Prettier separately.

Now, update your index.js again:

1module.exports = {
2 extends: [
3 'react-app', // or whatever you chose as your base config
4 'plugin:prettier/recommended', // this will stop Prettier and ESLint from fighting over fixes
5 'prettier/react', // optionally include this if you use React
6 ],
7 'plugins': ['prettier'],
8 'rules': {
9 // make Prettier return errors
10 'prettier/prettier': [
11 'error',
12 {
13 // Optional Prettier config changes
14 trailingComma: 'es5',
15 singleQuote: true,
16 printWidth: 80,
17 },
18 ],
19 }
20}

4. Publishing

To publish this config, you'll need to update a few things in your package.json:

1{
2 // must start with "eslint-config" for other people to include it
3 "name": "eslint-config-my-cool-config",
4 // this must be increased every time you publish
5 "version": "1.0.0",
6 // this tells node where to find the main module when it's installed
7 "main": "index.js",
8}

Then, make sure you have an account with NPM and follow their guide to publish a package.

Once it's successful, you should be able to go to any Javascript project using NPM and run:

1yarn add eslint-config-my-cool-config
2# or
3npm install --save eslint-config-my-cool-config

And then extend your own config by making a .eslintrc file in the root of your project:

1{
2 // don't include the `eslint-config-` part when extending
3 "extends": ["my-cool-config"]
4}

5. Super Charge Your Text Editor

If you aren't sold on the benefits of all this setup yet, this is where you'll start to see how helpful this is.

Your text editor should have an official ESLint integration, but for this example we're going to set it up in VS Code.

Install the VSCode ESLint extension and then open the command pallette (cmd / ctrl + shift + p) and type settings json and select Open Settings (JSON).

At the end of the settings JSON, add this:

1"editor.codeActionsOnSave": {
2 "source.fixAll.eslint": true
3},

Now when you save a file it should automatically run eslint fixes, which also include our Prettier formatting and save you hours of headache in the future.

Finally, don't forget that if you end up changing your mind about any rules, you can always override them in your .eslintrc project's config file:

1{
2 // don't include the `eslint-config-` part when extending
3 "extends": ["my-cool-config"],
4 "rules": {
5 // 0 turns the rule off, 1 emits a console warning and 2 throws an error
6 "overly-opinionated-rule": 0
7 }
8}

Or change them in your custom config package (just remember you'll have to publish it afterward):

1{
2 "extends": ["base-config"],
3 "rules": {
4 "overly-opinionated-rule": 0
5 }
6}
","scope":{"title":"Creating a Sane ESLint Config","description":"Stop wasting time fixing those semicolons and spaces","date":"2020-01-24","tags":["Programming","React"]}},"frontMatter":{"title":"Creating a Sane ESLint Config","description":"Stop wasting time fixing those semicolons and spaces","date":"2020-01-24","tags":["Programming","React"]}},"__N_SSG":true}