Better Programming

Advice for programmers.

Follow publication

The Case for pnpm Over npm or Yarn

Jacky Efendi
Better Programming
Published in
6 min readMar 19, 2020

pnpm is a package manager for JavaScript, like npm and yarn. I personally feel that pnpm is less known than it should be. According to the README in the repository, pnpm is:

• Fast. As fast as npm and Yarn.

• Efficient. One version of a package is saved only ever once on a disk.

Great for monorepos.

• Strict. A package can access only dependencies that are specified in its package.json.

• Deterministic. Has a lockfile called pnpm-lock.yaml.

• Works everywhere. Works on Windows, Linux and OS X.

This writing shares some of my learnings from migrating a monorepo (managed by a Yarn workspace + Lerna) that houses 60+ packages to pnpm. To understand more about why pnpm can be a good solution for your project, let’s start with a quiz.

Quiz

Imagine you’re starting a new JavaScript project. Like most people, you install your dependencies with npm or Yarn. Let’s use npm for this example. Your project needs Express, so you do:

npm install -g express

Dependencies installed!

Let’s say Express has a dependency package called debug.

How does your node_modules look now?

Try thinking about it for a few seconds …

.

.

.

.

.

Done?

If you answered A, you probably think that’s just how dependencies are supposed to work.

But if you try this with npm or Yarn, you’d find the actual answer is B, and this is a problem.

Why’s This a Problem?

With the structure shown in B, our code now can require('debug'), even though we don’t depend on it explicitly in our package.json.

Think about what will happen if:

  1. Express updates their debug dependency with breaking changes.
  2. Express decided to not depend on debug anymore.

In both cases, our code will now fail because it has an implicit dependency to debug.

The correct structure

In the correct structure, our code will never have access to debug. This is because of the way Node.js find stuffs inside node_modules.

Why Did npm Decide to Do This Then?

npm actually implemented structure A above before npm3, but there were some issues with it.

Duplications

If we add debug as our project dependency, we could have two different copies of debug.

The `debug` module is duplicated

Issues with this:

  • Duplicates in our disk
  • Possible duplicates in our bundles
  • Some packages break when there are duplicates (e.g., React)

Long-nested directories

Nested-directories hell

Some operating systems can’t handle long directories well.

Since version 3, npm has started to use flattened node_modules. You can read more about this here.

Flattened ‘node_modules’

Why’s This a Problem? (Continued)

We already established that having implicit dependencies isn’t ideal. If you work with a sizable monorepo, then the problem is even worse. It’ll be harder to trace where the actual dependencies a project uses came from.

Duplications are also an issue. Although Yarn does hoistings to optimise disk-space usage, it unexpectedly fails in some cases.

How pnpm Solves These

pnpm uses hard links and symlinks to achieve a semistrict node_modules structure and also to make sure only one version of a module is ever saved on a disk.

Let’s say we execute pnpm install express into our project. This is how our node_modules look like:

‘node_modules’ structure with pnpm

Notice that our code has no way to access debug because it’s not directly under the root node_modules directory.

pnpm creates a special .pnpm directory that contains all the modules’ hard links. Under express/4.0.0, there’s the express module, which is a hard link to the global pnpm-store, and a debug symlink to the debug hard link, which also links to the global pnpm-store.

The global ‘pnpm-store’

The global pnpm-store has a roughly similar structure. The actual package we install using pnpm is stored here. It’s normally saved under ~/.pnpm-store.

Huh, Does It Work With Every Package on the npm Registry?

Unfortunately, if you’re moving from Yarn/npm to pnpm, some packages might not work. Most of the time, this is caused by missing dependencies in the package’s package.json file.

Let’s use antd-table-infinity, for example. The package has no dependencies to antd but has some code that imports from antd. The package is also not bundling antd with it before being published. In most cases, when you’re using this package, you probably already also have antd installed. Then, the flattened node_modules structure allows this package to find antd. With pnpm, this package wouldn’t be able to find antd anymore and would fail.

Fortunately, pnpm provides hooks so we can resolve this kind of issue on our end. We can manually add antd as antd-table-infinity’s peerDependency using this simple hook.

simple ‘pnpmfile.js’ hook

How Does pnpm Benefit Us?

The two main benefits from pnpm, in my opinion, are:

No duplicates

  • There can only be one version of package on your machine
  • Saves disk space, no matter how many JS projects you have on your machine
  • No more issues with a duplicate React

Strict

  • No accidental access to nondependencies
  • More stable, more consistent, and more predictable
  • Avoid silly bugs

No duplicates mean no matter how many projects in your machine that depend on, let’s say, Electron, only one copy of each version of Electron will exist on your machine. It also work great for monorepos. No duplicates help in reducing the size of your monorepo on your disk. The strictness helps in catching hard-to-understand bugs earlier.

If you’re moving your project from npm or Yarn to pnpm, chances are you’ll find some unable to resolve module "some-module" errors. Your project might have worked previously because of the flattened node_modules structure, but will fail under the stricter structure — it’s a good thing if you’re aiming for long-term maintainability!

Summary

To summarise, pnpm is strict and helps us save disk space. The strictness makes things more predictable and catches bugs earlier. Whether that’s an important thing for you is up to you to decide. If you’re working with a large-scale monorepo, I highly recommend checking pnpm out.

This writing only covers some of what pnpm has to offer. Hopefully, you find it useful! More resources about pnpm can be found below. If you haven’t read through them yet, it’ll be worth it.

Sign up to discover human stories that deepen your understanding of the world.

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

Jacky Efendi
Jacky Efendi

Written by Jacky Efendi

I play around with JavaScript and all-things-web 🌐 | https://jackyef.com

Write a response

I think this needs to be updated to reflect yarn's Plug'n'Play functionality! It has actually existed for some time (2018 or earlier). Here are the docs: https://yarnpkg.com/features/pnp

--

"npm c set global-style true" can also achieve the same as pnpm.
... I think

--

This article popped up on my feed today, but after 9 months I don’t think it’s aged particularly well. Yarn 2 provides the same plug n play functionality that pnpm does.

--