ESLint: Linter Javascript

Revisión de errores o problemas


Aprender a programar en Javascript (cualquier lenguaje, en realidad) es una tarea que requiere ir perfeccionando continuamente. Una vez aprendemos las bases y fundamentos de Javascript, una estupenda recomendación sería la de usar un linter.

Los linters son herramientas que examinan tu código y te avisan cuando se detectan errores de sintaxis, código incorrecto o malas prácticas. Esto favorece el escribir código de calidad y acostumbra al usuario a solventar carencias o problemas en fases tempranas (y no tan tempranas) y ayudar a corregirlas.

Hay que tener claro que un linter nos ayuda a corregir malos hábitos que desembocan en problemas. Aunque un programa sea válido y funcione, puede estar escrito incorrectamente y ESLint te ayuda a detectar esos errores.

¿Qué es ESLint?

ESLint es un linter para código Javascript. Ayuda a detectar posibles problemas y errores (personalizables por el programador) para que el código Javascript escrito siga unos estándares de calidad, favoreciendo la escritura de código correcto, coherente y consistente. Aunque existen otros linters Javascript (JSHint, JSLint, etc...), ESLint se ha establecido como herramienta estándar de facto.

ESLint es totalmente configurable. Puedes indicarle que criterios (reglas) quieres que use y cuales quieres que ignore, adaptándose a las necesidades de cualquier empresa, programador o equipo de trabajo.

Instalación de ESLint

ESLint requiere como mínimo Node 18.18.0 o superior para funcionar. Siempre recomendamos tener la última versión estable de Node instalada, para lo cuál puedes seguir la Instalación con PNPM o la Instalación de NVM, un control de versiones que hará que la tarea sea muy cómoda y flexible.

El primer paso a realizar es instalar la herramienta ESLint es nuestro proyecto. Esto nos permitirá utilizarla para analizar nuestro código Javascript y detectar problemas. Para instalarla, escribiremos en una terminal el primer comando:

$ npm install -D eslint

$ npx eslint

Oops! Something went wrong! :(

ESLint: 9.13.0

ESLint couldn't find an eslint.config.(js|mjs|cjs) file.

Observa que tras instalar ESLint, intentamos ejecutar npx eslint pero obtenemos un error. Este error ocurre porque aún no tenemos fichero de configuración de ESLint, que es obligatorio para funcionar. Veamos como crear uno.

Configuración inicial de ESLint

Desde ESLint 9 se usa un formato llamado flat config que es mucho más rápido, extensible y simple. En este tutorial veremos como usar este formato, por lo que debes tener versiones superiores a la 9.

La forma más rápida de configurar ESLint es crear un fichero de configuración eslint.config.mjs en la carpeta raíz del proyecto. Este fichero contendrá, como mínimo, las siguientes líneas:

import globals from "globals";
import pluginJs from "@eslint/js";

export default [
  { languageOptions: { globals: globals.browser } },
  pluginJs.configs.recommended,
];

Ten en cuenta que debes instalar las dependencias anteriores con un npm install -D globals @eslint/js. Si todo esto se te hace muy complicado, puedes verificar la Configuración manual, donde el asistente de ESLint lo genera por ti.

El siguiente comando permite instalar y ejecutar el asistente @eslint/create-config de ESLint, que te preguntará varias cuestiones y creará un archivo de configuración.

$ npx eslint --init

Al arrancarse el asistente para configurar ESLint en nuestro proyecto, nos preguntará algunas cuestiones sobre el proyecto a revisar. Hagamos un repaso rápido por las opciones que nos aparecen:

Modo de ESLint

Esta opción nos permite seleccionar el modo en el que funcionará el linter. Se recomienda utilizar la última opción, la más completa de todas:

? How would you like to use ESLint?
  To check syntax only
> To check syntax and find problems

La primera opción, revisa sólo la sintaxis de nuestro código. La segunda opción (la recomendada), además ayuda a encontrar potenciales problemas en el código.

Módulos Javascript

Esta opción nos permite seleccionar la forma en la que importamos código Javascript desde otros archivos. Si estamos utilizando las palabras clave import y/o export en nuestro código, deberemos seleccionar la primera, ya que estamos utilizando módulos ESM. Si utilizamos una versión antigua de Node, o sigues utilizando require o module.exports, deberemos utilizar la segunda.

? What type of modules does your project use?
> JavaScript modules (import/export)
  CommonJS (require/exports)
  None of these

Si tienes más dudas, echa un vistazo al artículo Diferencias entre módulos ECMAScript (ESM) y CommonJS.

Framework Javascript

Esta opción nos da la posibilidad de indicar si estamos utilizando un framework Javascript específico como React o Vue.js y así poder adaptarse a la sintaxis correspondiente de estos frameworks. Si no fuera el caso, seleccionamos None of these:

? Which framework does your project use?
  React
  Vue.js
> None of these

Soporte de TypeScript

Si estamos transpilando nuestro código de Typescript a Javascript, podemos indicar en el linter que estamos utilizándolo. De lo contrario, simplemente indicamos que no lo utilizamos.

? Does your project use TypeScript?
> No
  Yes

¿Javascript cliente o servidor?

Con esta opción indicamos al linter si el código va dirigido a navegadores (browser) o es código NodeJS. Ojo con esta opción, ya que también podemos marcar ambas (utilizando SPACE):

? Where does your code run?
> Browser
> Node

Instalación de dependencias

Una vez respondidas estas preguntas, nos indicará algo similar a esto:

The config that you've selected requires the following dependencies:

eslint, globals, @eslint/js
? Would you like to install them now? ‣ No / Yes

Seleccionamos la opción Yes para instalar las dependencias. Al hacerlo, nos preguntará que gestor de paquetes quieres utilizar. La decisión más sencilla y habitual es npm, aunque personalmente me gusta mucho pnpm:

? Which package manager do you want to use? …
> npm
  yarn
  pnpm
  bun

Una vez terminado el proceso, generará un fichero de configuración llamado eslint.config.mjs.

Cómo usar ESLint

Una vez realizados los pasos anteriores, sólo tenemos que ejecutar eslint, indicando el archivo que queremos analizar. Por ejemplo, vamos a crear el siguiente archivo index.js con un código Javascript con ciertos problemas de sintaxis intencionados, a ver si es capaz de detectarlos:

let name;

const object = {
  name: "Manz",
  role: "streamer",
}

object = 34;

const showData = ()=>{
    console.log("Data")
  }

showData()

Ahora, escribiremos en una terminal el comando npx eslint seguido del archivo que queremos revisar para que nos avise de los errores que contiene nuestro código:

$ npx eslint index.js

/home/manz/project-test/index.js
  1:5  error  'name' is defined but never used             no-unused-vars
  8:1  error  'object' is constant                         no-const-assign
  8:1  error  'object' is assigned a value but never used  no-unused-vars

✖ 3 problems (3 errors, 0 warnings)

Nos ha encontrado tres problemas:

  • 💔 La variable name se define, pero no se usa nunca.
  • 💔 La constante object se define como constante, pero se intenta cambiar su valor.
  • 💔 La variable object se define, pero no se usa nunca.

También se pueden usar globs como *.js o similares. Uno muy interesante es ** que sirve para analizar toda la carpeta y subcarpetas. Por ejemplo, src/**/*.js buscará todos los ficheros .js dentro de la carpeta src y subcarpetas.

Corregir automáticamente (Autofix)

Una de las características más interesantes de ESLint, es utilizar la característica autofix, es decir, que ESLint no sólo detecte los errores, sino que los corrija automáticamente. Para ello, sólo tenemos que añadir el parámetro --fix:

$ npx eslint --fix index.js

Esto hará que ESLint revise el archivo y corrija todos los errores posibles de forma automática. Sin embargo, para nosotros lo ideal sería que nos lo corrija dentro de VSCode al guardar, en lugar de hacerlo desde una terminal.

Para conseguir esto, necesitamos «conectar» nuestro VSCode con ESLint:

  • Paso 1: Configurar ESLint ✅
  • Paso 2: Instalar el plugin ESLint en VSCode
  • Paso 3: Configurar VSCode para corregir cuando guardemos

Para realizar este último paso, simplemente pulsamos F1 en VSCode y y escribimos y seleccionamos la opción Open User Settings (JSON). En ese fichero .json escribimos lo siguiente:

/* ... */
"editor.codeActionsOnSave": {
  "source.fixAll.eslint": "always"
},
/* ... */

Este fragmento de configuración es el equivalente a escribir el comando anterior npx eslint --fix index.js al guardar el fichero. Recuerda que no todos los errores son autocorregibles, por lo que puede que ciertos errores no se arreglen de forma automática.

Personalizar reglas de ESLint

ESlint funciona con reglas. Las reglas son «criterios» para decidir si el código está haciendo algo correcto o incorrecto. Por defecto, ESLint viene con una serie de reglas definidas en su fichero de configuración. Por defecto usa recommended, pero podemos cambiar ese recommended por all e incluir todas las reglas:

import globals from "globals";
import pluginJs from "@eslint/js";

export default [
  {languageOptions: { globals: globals.browser }},
  pluginJs.configs.all,
];

Ahora, si ejecutamos nuevamente ESLint encontrará 7 errores, ya que tiene más reglas activadas y está detectando problemas más específicos:

$ npx eslint index.js

/home/manz/project-test/index.js
   1:5   error  Variable 'name' should be initialized on declaration  init-declarations
   1:5   error  'name' is defined but never used                      no-unused-vars
   8:1   error  'object' is constant                                  no-const-assign
   8:1   error  'object' is assigned a value but never used           no-unused-vars
   8:10  error  No magic number: 34                                   no-magic-numbers
  10:1   error  Combine this with the previous 'const' statement      one-var
  11:5   error  Unexpected console statement                          no-console

✖ 7 problems (7 errors, 0 warnings)

Es decisión del desarrollador definir que nivel o grado utilizar para revisar su código. Observa que a la derecha del error, aparece un texto con el nombre de la regla (init-declarations, no-unused-vars, etc...). Cada una de esas reglas define el comportamiento de una característica. Puedes consultar las reglas existentes en ESLint rules.

Puedes configurar en el fichero de configuración el comportamiento de las reglas para ajustarlo a tu gusto. Vamos a crear un objeto rules con las reglas que queremos activar y desactivar:

import globals from "globals";
import pluginJs from "@eslint/js";

const rules = {
  "no-unused-vars": "off",
  "no-magic-numbers": "warn"
};

export default [
  {languageOptions: { globals: globals.browser }},
  pluginJs.configs.all,
  { rules }
];

Simplemente colocamos el nombre de la regla y en su valor off para desactivarla, warn para avisar pero no marcarlo como error y error si queremos que lo detecte como error.

Finalmente, añadimos { rules } al array del export default. Asegúrate de ponerlo al final, para que sobreescriba a las anteriores, si existieran.

Podemos utilizar npx eslint --inspect-config para lanzar un asistente que te enviará al ESLint Config Inspector, una web local donde verás cómodamente las reglas activadas, tu configuración...

Usar un pack de reglas

Esto puede ser muy laborioso y lento para mucha gente, por lo que lo ideal suele ser instalar un paquete de reglas para ESLint. Hay muchos y muy variados, donde su creador ha añadido las reglas que ha considerado oportunas para su forma de trabajar.

Veamos algunos packs populares de reglas de ESLint:

Nombre del paquete (NPM)PromotoresGitHub
eslint-config-airbnbAirbnb@airbnb/javascript
neostandardNeostandardneostandard/neostandard
eslint-config-xoXO@xojs/eslint-config-xo

Veamos un ejemplo de instalación de neostandard, que es una evolución del clásico StandardJS. En primer lugar, instalamos el pack de reglas:

$ npm install -D neostandard

Una vez instalado, modificamos el fichero de configuración eslint.config.mjs y añadimos el pack:

import globals from "globals";
import pluginJs from "@eslint/js";
import neostandard from "neostandard";

export default [
  {languageOptions: { globals: globals.browser }},
  pluginJs.configs.all,
  ...neostandard()
];

Observa que hemos importado el paquete neostandard y lo añadimos al array de configuración mediante ...neostandard(), ya que es una función que devuelve un array. Con esto ya lo tendríamos. Ahora, ejecutamos el linter:

$ npx eslint index.js

/home/manz/project-test/index.js
   1:5   error  Variable 'name' should be initialized         init-declarations
   1:5   error  'name' is defined but never used              no-unused-vars
   1:9   error  Extra semicolon                               @stylistic/semi
   4:9   error  Strings must use singlequote                  @stylistic/quotes
   5:9   error  Strings must use singlequote                  @stylistic/quotes
   8:1   error  'object' is constant                          no-const-assign
   8:1   error  'object' is assigned a value but never used   no-unused-vars
   8:10  error  No magic number: 34                           no-magic-numbers
   8:12  error  Extra semicolon                               @stylistic/semi
  10:19  error  Missing space before =>                       @stylistic/arrow-spacing
  10:22  error  Missing space after =>                        @stylistic/arrow-spacing
  11:1   error  Expected indentation of 2 spaces but found 4  @stylistic/indent
  11:5   error  Unexpected console statement                  no-console
  11:17  error  Strings must use singlequote                  @stylistic/quotes
  12:1   error  Expected indentation of 0 spaces but found 2  @stylistic/indent

✖ 15 problems (15 errors, 0 warnings)
  9 errors and 0 warnings potentially fixable with the `--fix` option.

Observa que muchos de los errores tienen una regla prefijada por @stylistic/. Esto son problemas de sintaxis que tienen que ver con el formateo del código:

  • ✨ Falta un punto y coma
  • ✨ Indentaciones incorrectas
  • ✨ Falta de espacios entre parámetros
  • ✨ Comillas simples en vez de dobles
  • ✨ etc...

Por lo tanto, ahora no sólo busca problemas de código sino también de formateo, que combinado con el autofix, puede funcionar de una forma similar a Prettier, pero siendo muy configurable.

Alternativas a ESLint

Ten en cuenta que en esta guía hemos visto como configurar ESLint con VSCode. Sin embargo, también es posible configurarlo en otros editores, a través de plugins como ESLint para SublimeText, ESLint para IntelliJ IDEA, ESLint para Rollup o ESLint para vim.

Por otro lado, aunque ESLint es probablemente la mejor y más completa herramienta para revisar código Javascript, existen algunas alternativas que podrían resultar interesantes en algunos escenarios:

HerramientaDescripciónWeb
Oxidation CompilerLinter rápido basado en Rust, compatible con ESLint.oxc-project/oxc
BiomeLinter rápido basado en Rust, compatible con Prettier.biomejs/biome
PrettierFormateado opinionado. Es cómodo, pero poco personalizable.prettier/prettier

Si estás buscando alternativas, quizás una de las anteriores te pueda servir.

¿Quién soy yo?

Soy Manz, vivo en Tenerife (España) y soy streamer partner en Twitch y profesor. Me apasiona el universo de la programación web, el diseño y desarrollo web y la tecnología en general. Aunque soy full-stack, mi pasión es el front-end, la terminal y crear cosas divertidas y locas.

Puedes encontrar más sobre mi en Manz.dev