I have an npm package with a fixed version that has an update.
Example package.json extract:
devDependencies: {
"someFixedVersionPackage": "1.0.0", //1.1.0 is latest
"anotherFixedVersionPackage": "2.3.2", //2.3.4 is latest
}
Does an npm command exist which installs the latest version of that package and updates the package.json, preferably all packages at once?
To be clear, I want the package.json snippet above to be updated to this, in addition to the packages themselves being updated:
devDependencies: {
"someFixedVersionPackage": "1.1.0", //latest
"anotherFixedVersionPackage": "2.3.4", //latest
}
Thank you.
Why doesn't
npm update
work here?As per the documentation on
npm update
:Since your packages are defined with a fixed version, the update sub-command will not update those to respect semantic versioning. Therefore, it will only automatically update your packages if you specify a greater version range for each package. Note that it is actually typical in an npm project to specify a loose range version; one that is meant to avoid breaking changes but still leaves room for improvements and fixes.
Still, why shouldn't I fix dependency versions in my package.json?
Having a list of dependencies with a fixed version does not mean that the dependencies installed will always be the same, because the dependencies of your dependencies will most likely also be defined with a version range. In order to keep track of a list of tested version-tagged dependencies, npm provides another mechanism: package locks.
Before version 5 of npm, you can create a "npm-shrinkwrap.json" file with the
shrinkwrap
command:Since npm 5, a "package-lock.json" is automatically generated when an npm operation modifies the "node_modules" tree or "package.json".
Rather than modifying package.json, either one of these package locks will override the default behaviour of
npm install
, installing dependencies with the versions specified by the lock, right when they were created or manually updated. With that out of the way, your dependencies can now be expanded without the risk of dependents installing untested package versions.Shrinkwraps are used for publishing packages. To shrinkwrap a package:
npm install
in the package root to install the current versions of all dependencies.npm shrinkwrap
, add npm-shrinkwrap.json to git, and publish your package.At this point, dependency versions can be loosened in your package.json (this will hopefully be done only once every major dependency update), so that later on they can be updated at will with
npm update
:The package-lock.json file can be used instead of a shrinkwrap, and is more suitable for reproducing a development environment. It should also be committed to the repository.
So how do I update my dependencies?
Calling
npm update
will do what's mentioned above: update dependencies while respecting semantic versioning. To add or upgrade a dependency in a package:npm install
in the package root to install the current versions of all dependencies.npm install --save
each new or updated package individually to update the package.json, as well as the existing package locks ("package-lock.json" and "npm-shrinkwrap.json"). Note that they must be explicitly named in order to be installed: runningnpm install
with no arguments will merely reproduce the locked dependencies.Moreover, here are a few tips for a smooth transition from a project with fixed dependencies:
If you haven't done so, expand the version range by adding a tilde (
~
) before the version specifier, or a caret (^
).npm update
will then attempt to install all patch revisions and minor revisions, respectively (major version0
is a corner-case, see the documentation). For instance, "^1.0.0" can now be updated to "^1.1.0", and "~2.3.2" can be updated to "~2.3.4". Adding the--save
or--save-dev
flags will also update the "package.json" with the installed version (while keeping the previous range specifiers).Run
npm outdated
to check which packages are outdated. Entries in red will be updated automatically withnpm update
. Other entries will require a manual intervention.For packages with major version bumps, install that package with a version specification (e.g.
npm install browserify@11.2.0 --save-dev
). Further issues that may arise with the update will have to be handled manually. It usually helps to read the news feed or the release history on that package to further understand what has changed from previous versions.This is not simple enough, is there another way to do this?
Before continuing, it is always worth mentioning that packages have a SemVer-compliant version definition for a reason. One should avoid blindly installing the latest version of every single package. Although such a full update can be done and tools are available for that, some caution is advised. For instance, you would not want to install React 15 if the remaining React components and libraries are not compatible with
react@15.x.x
. See also npm's blog post: Why use SemVer?I'll take my chances. What other tools are there?
To name a few:
npm-check-updates
will do what was initially asked in the question: install and update the versions of all dependencies, regardless of the given range constraint. This would be the least recommended tool for the job, however.updtr
will update dependencies one by one and roll back to the previous version if the project's tests fail, which may save time in projects with good test coverage.npm-check
provides an interactive command-line interface, which allows you to easily select which packages to update.Is this any different with npm 5?
Since major version 5, npm will automatically create a "package-lock.json", which will fill the role of specifying the dependency tree when a shrinkwrap does not exist. A more detailed description can be found in the package-locks documentation. In general, npm-shrinkwrap.json is meant to be used when publishing, whereas package-lock.json is to be used in development. This is why you should also commit "package-lock.json" to the repository.
What about with Yarn?
Yarn, an npm-compatible dependency manager, creates a lock file automatically on use, which behaves similarly to the npm shrinkwrap. Calling
yarn upgrade «package»
will update one dependency to the version in thelatest
tag, regardless of the version range recorded in the package.json or the lock file. Usingyarn upgrade-interactive
also allows you to selectively upgrade packages to the latest version, not unlikenpm-check
.Running the following command will do what you want:
Breakdown:
npm install someFixedVersionPackage@latest
will install the latest version of the package--save-dev
flag will cause it to update the version in yourpackage.json
'sdevDependencies
--save-exact
flag will cause it to save a fixed version instead of a semver range operatorLink to the npm install docs
I've been looking for an easy way to update npm dependencies for a long time. Then I found this tool: https://github.com/dylang/npm-check
It shows you which dependencies are out of date in a nice ui and allows you to update them. It even tells you which ones are likely to break due to major changes and warns you of unused dependencies.