[BUG] `npm install` sometimes removes indirect dependencies if a parent node was deleted from the lockfile
Is there an existing issue for this?
- [X] I have searched the existing issues
This issue exists in the latest npm version
- [X] I am using the latest npm
Current Behavior
My team sometimes deletes nodes from package-lock.json as specific package versions are deleted from our internal registry. Our expectation is that npm install should then install the latest version of that package that satisfies the requirements in package.json, without changing indirect dependencies unnecessarily.
However, we're seeing an issue where npm install sometimes deletes an indirect dependency of the deleted node, even as it replaces that node. We have to run npm install a second time in order to restore the indirect dependency.
This issue only seems to occur when there's a different version of the indirect dependency installed.
Expected Behavior
We expect package-lock.json to always be in a consistent state after an npm install, with all dependencies satisfied.
Steps To Reproduce
I've created a CodeSandbox: https://codesandbox.io/p/devbox/quirky-rain-rv9lkl You can create a new project in the same state by running npm install [email protected] && npm install [email protected] && npm install [email protected].
Once you have that project set up, the steps to replicate the bug are:
- Delete the
"node_modules/mocha"node frompackage-lock.json. - Run
npm install. - Check the diff and see that npm restored the
"node_modules/mocha"node, but removed the"node_modules/mocha/node_modules/brace-expansion"node. That directory has also been deleted fromnode_modules. This means that mocha's indirect dependency onbrace-expansion@^2.0.1(by way of its dependency onminimatch@^5.1.6) is unsatisfied; mocha would instead use[email protected], which is installed at the root ofnode_modules. - Run
npm installagain and observe that"node_modules/mocha/node_modules/brace-expansion"has been restored in bothpackage-lock.jsonandnode_modules.
Environment
I've observed this issue in npm v8, v9, and v10.
- npm: 10.8.2
- Node.js: 20.16.0
- OS Name: macOS 14.6.1
- System Model Name: M1 MacBook Pro
- npm config:
; node bin location = /Users/trevorburnham/.asdf/installs/nodejs/20.16.0/bin/node
; node version = v20.16.0
; npm local prefix = /Users/trevorburnham/Code/lockfile-with-missing-parent-testcase
; npm version = 10.8.2
; cwd = /Users/trevorburnham/Code/lockfile-with-missing-parent-testcase
; HOME = /Users/trevorburnham
I've put together an arborist test case to demonstrate the issue: https://github.com/TrevorBurnham/cli/tree/7746-bug-test-case
You can run the test case on that branch with this command:
npm test -w @npmcli/arborist -- test/arborist/reify.js --no-cov -g="parent"
Notably, the test passes if you perform reify twice: The second reify restores the indirect dependency that the first reify deleted,
I've submitted a PR that appears to address this issue (at least as far as the test case is concerned): https://github.com/npm/cli/pull/7752
I'd welcome alternative fix suggestions. I'm new to this codebase, but from what I can tell, here's what's happening when building the ideal tree in the test case:
- The dependency on the deleted node is identified as a problem edge and the information needed to recreate the node is fetched from the registry.
- When the new node is placed on its parent, that edge is now considered satisfied. And since there are already nodes for all of its direct dependencies, all of its edges out are also considered satisfied (i.e. it has no problem edges).
So the problem is that no checks are performed for indirect dependencies of the placed dep. The PR addresses that by adding the placed dep's children to the deps queue, so that they get checked for missing deps.
I'm sure there's a more elegant solution that I'm missing! I'd love to hear any thoughts.