From 5f1776971f35620803a588c8e4549474b3f9d6f3 Mon Sep 17 00:00:00 2001
From: Eugenia <42525512+meugeniatr@users.noreply.github.com>
Date: Tue, 22 Oct 2024 09:41:10 +0000
Subject: [PATCH 1/4] Article : How to use ESLint to improve your workflow
---
.../2024-25-10-use-eslint-improve-workflow.md | 283 ++++++++++++++++++
_assets/authors/meugeniatr.jpeg | Bin 0 -> 176224 bytes
_assets/authors/meugeniatr.jpg | Bin 20190 -> 0 bytes
3 files changed, 283 insertions(+)
create mode 100644 _articles/en/2024-25-10-use-eslint-improve-workflow.md
create mode 100644 _assets/authors/meugeniatr.jpeg
delete mode 100644 _assets/authors/meugeniatr.jpg
diff --git a/_articles/en/2024-25-10-use-eslint-improve-workflow.md b/_articles/en/2024-25-10-use-eslint-improve-workflow.md
new file mode 100644
index 000000000..b784ef051
--- /dev/null
+++ b/_articles/en/2024-25-10-use-eslint-improve-workflow.md
@@ -0,0 +1,283 @@
+---
+contentType: article
+lang: en
+date: '2024-25-10'
+slug: use-eslint-improve-workflow
+title: How to use ESLint to improve your workflow
+excerpt: >-
+ ESLint ecosystem can be super handy for JS and TS codebases, but are you using it right? Keep reading to learn about useful ESLint rules and a get a little bit more out it.
+categories: []
+authors:
+ - meugeniatr
+keywords:
+ - javascript
+ - typescript
+---
+
+## A few words about my experience using ESLint
+I have to confess that ESLint and I haven't always been on the best of terms. My first steps with ESLint have been shy. I stuck to the the default setup within my first projects. Over time, I became more familiar with it, but only by struggling in my merge requests when unexpected updates from colleagues would cause issues in the repository.
+
+It might seem obvious, but there are many linters available that suit different projects and languages, such as [checkstyle](https://github.com/checkstyle/checkstyle) for Java, among others. In this article, I'll focus on ESLint for TypeScript and React codebases.
+
+For a long time, ESLint felt distant to me, more like a satellite hovering around the codebase. While I had read about more advanced configurations, it didn't quite make enough noise to grab my attention. However, my interest slowly but surely grew in this widely popular tool. A turning point was attending the [JS Nation conference in Amsterdam](https://jsnation.com/) last summer, where Anthony Fu gave an insightful talk titled [ESLint One for All Made Easy](https://gitnation.com/contents/eslint-one-for-all-made-easy). His talk, which covered the new flat config feature introduced in ESLint 9, opened my eyes to how ESLint could be more versatile and customizable, especially for monorepos. I'll pick a little bit of this talk in this article, but I highly recommend watching his talk for a more detailed explanation. Anthony is fantastic!
+
+## What is ESLint and why people use it?
+ESLint was created by [Nicholas C. Zakas](https://github.com/nzakas) in 2013 as a successor to JSLint (Douglas Crockford, 2002) and its fork, JSHint (Anton Kovalyov, 2011). Like its predecessors, ESLint was designed for linting JavaScript projects, but with much greater flexibility and customization.
+
+
What is linting?
+_A *linter* is a static analysis tool used to identify and prevent bugs, stylistic errors, or unconventional code patterns. The term "lint" originally comes from the world of fabrics, referring to the small fibers that break off from tissues, causing issues for machines and making clothes look uneven or worn. In software development, a linter helps keep our large code sheet "clean," much like removing lint helps keep fabrics smooth._
+
+_The concept of linting in computer science was first introduced in 1978 with a Unix utility for analyzing C code, created by Stephen C. Johnson during his debugging process._
+
+
+
+While JSLint was a very opinionated linting tool, it was also groundbreaking. It opened a door to a different view on debugging, prioritizing code quality through static analysis and compilation optimizations. ESLint expanded on this concept, gaining popularity thanks to its highly customizable rule set and flexibility, which allowed developers to adapt it to various coding standards and environments. This versatility helped ESLint grow into the robust tool with a thriving community that it is today.
+
+## How can ESLint improve your codebase?
+ESLint is a very common tool for linting languages before the compilation phase. This means that developers can catch errors and identify styling rules in an earlier process than in a code review. And this is honestly quite useful for detecting technical issues and having a consistent codebase. All these can seem pretty obvious after explaining what a linter does. But I would like to share you how eslint can help not only your repository, but also your team.
+
+Reviewing a merge request can become a problematic process when the comments circle around different stylistics points. This is when discussing ESLint rules can be a big relief for the team, since everyone can share their point of view about writing code and -hopefully- agree on a rule that will save time and frustration when reviewing.
+
+Well, this is already a big win but there's more. The same goes about code quality: linting can be a great objective tool to measure it and cut down some long exchanges about which way is better in technical reviews. This can also be a great help to junior developers who are still learning good practices or adjusting in the onboarding process of a codebase.
+
+### Basic React and Typescript set-up
+In any React and Typescript project, you have already installed and set up [typescript-eslint](https://github.com/typescript-eslint/typescript-eslint), [eslint-plugin-react-hooks](https://www.npmjs.com/package/eslint-plugin-react-hooks)
+ and [eslint-plugin-react](https://www.npmjs.com/package/eslint-plugin-react). If you didn't, then you should. This article 'takes for granted' that you are already using both of these tools.
+
+### A few helpful rules
+I have used ESLint for JS and TS projects, and in these experiences I found that this tool can avoid numerous comments in merge requests and upset dev teams. We have gathered a set of rules regarding the topics that kept coming back during the review process and discussed whether implementing them or not in technical meetings. The process has gone smoothly since then. Comments regarding style and conventions dropped dramatically, and it helped newcomers to adapt their writing to the repository too.
+
+I think it has been a huge step forward, since it has also helped us to catch bugs and issues with types even if we already use TypesScript. These are the rules that, for me, have become essential:
+
+- ### Sorting types, objects, props
+
+Sorting is tricky, but many times it can be pertinent to improve readability. So we adopted a bunch of rules to automatize this process:
+
+
+ - [eslint-plugin-sort-destructure-keys](https://www.npmjs.com/package/eslint-plugin-sort-destructure-keys)
+
+Maybe the easiest win, with only one option to customize (caseSensitive), this rule will sort every destructured object:
+
+```js
+/* caseSenditive true */
+let { a, B, c } = obj; 🚫
+let { B, a, c } = obj; ✅
+```
+
+- [perfectionist/sort-object-types](https://perfectionist.dev/rules/sort-object-types.html)
+
+Perfectionist plugin can be mind blowing! This particular rule allows many options. I particularly like the fact that it allows types to be sorted by _required_ first, as well as being able to pass an array with particular keys that we want to see sorted first, such as "key" or "id".
+
+```ts
+/* before 🚫 */
+type Department = {
+ location: string
+ departmentName?: string
+ employees: number
+ head?: string
+ established?: Date
+ id: string
+}
+
+/* after, using requiredFirst true and customGroups ✅*/
+type Department = {
+ id: string
+ location: string
+ departmentName?: string
+ employees: number
+ established?: Date
+ head?: string
+}
+```
+- [react/jsx-sort-props](https://github.com/jsx-eslint/eslint-plugin-react/blob/master/docs/rules/jsx-sort-props.md)
+
+Straigh from eslint-plugin-react, this rule sorts all props alphabetically... automatically! It has plenty of cool options for sorting callbacks, shorthands, etc. This is my favorite config:
+
+```js
+'react/jsx-sort-props': [
+ 'error',
+ {
+ callbacksLast: true,
+ shorthandFirst: true,
+ ignoreCase: true,
+ noSortAlphabetically: false,
+ reservedFirst: true,
+ },
+],
+```
+
+This will update your code like this:
+
+```js
+/* before 🚫 */
+
+
+/* after ✅*/
+
+```
+
+### Code correctness
+
+#### [typescript-eslint/no-unnecessary-condition](https://typescript-eslint.io/rules/no-unnecessary-condition/)
+
+Any time a boolean expression always evaluates to the same value, this rule will catch it. It helps you remove unnecessary expressions that use the `??` operator, `?` operator and `if` conditions. ESLint will evaluate code that precedes an expression and it will deduce the correct type.
+
+The following example shows the expression catching a value that will never be `undefined` so we don't need to check for it.
+
+```ts
+const obj: {value?: string;} = {}
+obj.value = 'test';
+// Unnecessary optional chain on a non-nullish value.
+console.log(obj.value?.length)
+// Unnecessary conditional, expected left-hand side of `??` operator to be possibly null or undefined
+console.log(obj.value.length ?? 0); 🚫
+```
+
+In the same manner, it can detect unnecessary conditions:
+
+```js
+const answer = 42;
+// Unnecessary conditional, value is always falsy.
+if (answer < 0) { 🚫
+ console.log("Never happens");
+}
+```
+
+#### [no-floating-promises/](https://typescript-eslint.io/rules/no-floating-promises/)
+
+Unhandled (or wrongly handled), promises can lead to race conditions and unexpected behaviour.
+
+This rule enforces that all promises have to be awaited, returned, voided, or have their `then()` or `catch()` function called.
+
+Examples of bad promise handling:
+
+```ts
+// not awaited 🚫
+const promise = new Promise((resolve, reject) => resolve('value'));
+promise;
+
+// Then called with a single parameter. 🚫
+async function returnsPromise() {
+ return 'value';
+}
+returnsPromise().then(() => {});
+
+// Catch called with no parameter. 🚫
+Promise.reject('value').catch();
+
+// No then or catch.
+Promise.reject('value').finally();
+
+// Async called within a map instead of Promise.all 🚫
+[1, 2, 3].map(async x => x + 1);
+```
+
+#### [require-await](https://typescript-eslint.io/rules/require-await/)
+
+This rule checks for functions that are marked async that don't either return a promise or await a function in their code.
+
+
+```ts
+// Does not return a promise. 🚫
+async function returnNumber() {
+ return 1;
+}
+```
+
+### Complexity
+Simpler code is easier to read and maintain. ESLint community has developped many rules to help this happen. If your codebase does not use these rules yet, I advice you to add them as warnings and progressively refactor sections of the code.
+
+#### [no-uneeded-ternary](https://eslint.org/docs/latest/rules/no-unneeded-ternary)
+
+Some developers love their ternary operators. These rules will keep them in check.
+
+The common mistake is to leave code like this after refactoring.
+
+```ts
+/* before 🚫 */
+return answer ? true : false;
+```
+
+```ts
+/* after ✅*/
+return answer;
+```
+
+Another superfluous code this rule will find is when a ternary could be replaced by a simpler expression:
+
+```ts
+/* before 🚫 */
+foo(bar ? bar : 1);
+
+/* after ✅*/
+foo(bar || 1)
+```
+
+#### [no-nested-ternary](https://eslint.org/docs/latest/rules/no-nested-ternary)
+
+Nested get nasty. Nested ternaries are illegible so they should be avoided. Activating this rule will forbid them.
+
+
+#### [complexity](https://eslint.org/docs/latest/rules/complexity)
+
+Code should be either "long" or "wide", if a function contains nested conditions, there should be only one of them, if it is long, it should only contain simple conditions.
+
+The measure of "cyclomatic complexity" expresses the complexity of code in terms of possible branches.
+
+```ts
+// Complexity: 1
+return calculate();
+
+// Complexity: 2
+// There are two possible paths through this code depending if foo is truthy or falsy.
+if (foo) {
+ return 0;
+}
+return calculate();
+
+// Complexity: 3
+if (foo) {
+ return 0;
+}
+
+// Additionally there are two additional paths here depending on the value of bar.
+return bar || calculate();
+```
+
+You can use the rule to set a maximum allowed value for the complexity (by default it is 20). Setting this to a lower value will force the developers in your team to write shorter and simpler functions.
+
+Setting this too low might interfere with functions that are just long lists of if clauses (like routers).
+
+### Config alternatives
+Maybe you have already heard about [epicweb](https://www.epicweb.dev/), but in case you didn't, it is a platform created by [Kent C. Dodds](kentcdodds.com) among other very talented tech people where you can learn a lot. And I mean a lot! There's workshops, articles, tips.
+
+But coming back to this article's topic, they have also developped a basic rules config to get get started with ESLing, Typescript and Prettier. You can check it out [here](https://github.com/epicweb-dev/config.) Even if the same set-up can be done by ourselves, these rules working together make sense and they even [took the time to write an article about their chosen Prettier rules](https://www.epicweb.dev/your-code-style-does-matter-actually). It can be really inspiring!
+
+Another approach has been presented by Anthony Fu during JS Nation Amsterdam. He pushes ESLint rules not only as a linter but also as a formatter, getting rid of Prettier configurations in his personal ESLint config https://github.com/antfu/eslint-config. Support for React, Svelte, UnoCSS, Astro, Solid is provided, as well as the newly released ins ESLint 9 flat-config.
+
+## Conclusion
+Linter is a great tool for improving code. It is worth that everyone in the team gets involved with its configuration for smoothing the review process and code quality. Even if linters do not replace a review, they quite help brush off a chunk of comments and helps speed up the delivery of properly conventioned code.
+I like to think that this is a way of democratizing code within a team, as well as improving it. But this also means taking time for exploring and testing. If you made it up to here, thanks! I hope you are a bit more ready to play around ESLint!
+
+## Resources and further digging
+- ESLint official page [https://eslint.org/](https://eslint.org/)
+- ESLint on Wikipedia [https://en.wikipedia.org/wiki/ESLint](https://en.wikipedia.org/wiki/ESLint)
+- ESLint One for All Made Easy - Anthony Fu https://gitnation.com/contents/eslint-one-for-all-made-easy
+- How to Use Linters and Code Formatters in Your Projects - German Cocca https://www.freecodecamp.org/news/using-prettier-and-jslint/
+- Top Ten ESlint Rules for Any TypeScript Codebase - Kevin Schaffter https://blog.stackademic.com/top-ten-lint-rules-for-any-typescript-codebase-cb3148e67aca
+- What is Linting? How does a linter work? - Dogan Ogut [https://ogutdgnn.medium.com/what-is-linting-how-does-a-linter-work-49381f28fc60](https://ogutdgnn.medium.com/what-is-linting-how-does-a-linter-work-49381f28fc60)
+- Easing into Cyclomatic Complexity - Peter Perlepes https://dev.to/igneel64/easing-into-cyclomatic-complexity-38b1
\ No newline at end of file
diff --git a/_assets/authors/meugeniatr.jpeg b/_assets/authors/meugeniatr.jpeg
new file mode 100644
index 0000000000000000000000000000000000000000..966a33532eb917e44c05b9214862d5616e2d1046
GIT binary patch
literal 176224
zcmb4~WlSYptgdl)cNlbVcXxMpcXx+jaCdiyjl09(&c@ws<1o0-{mw~l?(h4y|C+Uu
zCQZ{OPyem|+Xq9FlaiGJ0|y5K1OLy!{%wJYgF!+-{Ez=9sQ(-q4jLK?3K{_x76uLp
z0SO5a0TB@y1p^Hk1sw$u5e)|o9TN*18yg7~7Y_#u4+9Gu>wi80hy3pi6f`_EG&~kE
zA~M$hxBd-*p~FFv{b%6hU=Zlwkm%t5M!<-`z#t&O{}b5%6(ke{Gz>TxEZl!lLU6GE
z4gS9<7&yd#8Nk5)+XO>`1P6magGBo;)85wNzK6i?S9wvp6;%S(6@y$MwW0LG^h(6&
z70KCLxnjD%+r_kJqxo<)6cT2wEaeZm!yt6`Mi}X#?P)MkXNqVO&Vv~RKcvA_YQc4|yenB)2AYyXlxCT~A%~q|MLE
zz%rqZmL@OFDF7pNEPct6u>`|hP+mZ5CyLb~A0LKeYl-Cjrq)vhg6mk7t$Qz%sQdPX(t
zu_>=VM56Q{>P(h7lj!yM2~c@=hS4iED`gZJ?S{79R(6&6nChL#q>eQid6VpXn?{kX
zlC+P3%LhQOG@e;Uayh8Euo?k^BnGm-5IiO7?c>Z`yt#mcN50uWlpJxX(alb+ogr+r
z2I?u=U6xhp?p5`v>Sn{*vZu+e%BOOGPLCpLq*y_t!emb{OVLfpZ?|R#EVzq7
!d;si5C3Q(?}^?)#Va
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