This is a question about plugin modules resolution in ESLint, when using pnpm. There are many articles and answers about this online, I have read most of it.
I have also educated myself on pnpm's node_modules structure and ESLint's resolution algorithms. I still do not understand why we do not need to install the plugins used under the hood by the @vercel/style-guide
module.
This is what the ESLint docs says about plugin resolution:
Plugins are resolved relative to the config file. In other words, ESLint loads the plugin as a user would obtain by running
require('eslint-plugin-pluginname')
in the config file.Plugins in the base configuration (loaded by extends setting) are relative to the derived config file. For example, if
./.eslintrc
hasextends: ["foo"]
and theeslint-config-foo
hasplugins: ["bar"]
, ESLint finds theeslint-plugin-bar
from./node_modules/
(rather than./node_modules/eslint-config-foo/node_modules/
) or ancestor directories. Thus every plugin in the config file and base configurations is resolved uniquely.
@vercel/style-guide
exposes some ESLint configs, like the react.js config, which in turn will extend: plugin:eslint-plugin-react
, which in turn contains plugin: { react }
.
With that in mind, it is safe to assume that when our application extends @vercel/style-guide/react
is should be necessary to install the plugins it uses, i.e. eslint-plugin-react
directly as a dependency of the end user app.
I tried it, and for some reason it is not necessary to install them.
Why is that? What am I not seeing about the resolution of these plugins?
I found the reason it works without manually hoisting those packages.
The answer was that pnpm's default setting for
public-hoist-pattern
is:['*eslint*', '*prettier*']
As such, it is crucial not to overwrite that setting. If changing it, include the defaults as well. For instance:
EDIT
I found out that
@vercel/style-guide
uses a community package that changes the behavior of how ESLint resolves plugins, so that it no longer resolves them from the derived config, but instead from the config where the plugin is actually mentioned.The Vercel team clarified it here: https://github.com/vercel/style-guide/issues/86#issuecomment-1924675205
The package: https://www.npmjs.com/package/@rushstack/eslint-patch