Skip to content

Latest commit

 

History

History
91 lines (54 loc) · 7.6 KB

dependency-management-guidelines.md

File metadata and controls

91 lines (54 loc) · 7.6 KB

Dependency management

What

This document outlines the best practices on keeping package dependencies (including Node itself) up to date.

Why

One of the basic maintenance tasks is ensuring that your package uses the latest stable versions of its dependencies. There are multiple benefits of doing this:

  • The primary reason to keep your dependencies up to date is to receive all the latest bugfixes and improvements.
  • Old versions of your dependencies might no longer be receiving security updates, which means that if a new serious vulnerability is disclosed, you may be forced to do extra unplanned work if you are multiple versions behind.
  • Upgrading regularly in small incremental steps is potentially less disruptive than doing a big bang update to adapt to many breaking changes released over multiple versions of a dependency.

There are some potential caveats to take into consideration when upgrading dependencies:

  • Dropping support for specific Node.js versions is normally considered a breaking change, which means that upgrading a dependency might imply that you also need to drop support for these versions, which in turn will force you to release a new major version of your package as well.
  • Upgrading a dependency could result in increased size of the end user's node_modules / compiled bundle, if other packages in the ecosystem do not upgrade around the same time.

Node.js itself can also be considered a dependency of your package, therefore you should make sure to test in latest LTS and current Node.js versions in your Continuous Integration (CI) system - see Choosing the Node.js version for testing.

How

Choosing dependencies

There is no single recipe for choosing the right dependency for your package, but there are some indicators to pay attention to and questions to ask yourself:

  • Should you build it yourself or use a package from the ecosystem? Adding a dependency is a trade-off. Each dependency has a download cost, and if it is built by someone else - it may have security and reliability implications. However building your own implementation may suffer from even worse bugs (incl. security) and might not have enough eyes to catch them.
  • Does the package have tests and do they run automatically in a CI system, e.g. Travis or GitHub Actions.
  • Is the package actively maintained? Lack of activity in the source control does not mean it is unmaintained, as it may simply be done, but it is something to pay attention to.
  • Make sure the package is not already marked as deprecated on npm and in the README - authors may give you suggestions for alternatives.
  • Does the project offer support? Do the releases have a changelog to ease the update?

Defining the dependency version ranges

The JavaScript ecosystem relies on semantic versioning and most packages respect its rules. Therefore, it is usually recommended that your package.json accepts new minor/patch releases of your dependencies automatically. This is the default behavior of npm - when you first npm install [new dependency], it will by default prefix the version with a ^ to indicate that all new releases under the same major version are acceptable.

You may also use a tilde (~) character instead of the caret (^), to only accept bugfix (patch) releases, but this may mean that your dependency will become out of date sooner.

Accepting all releases (i.e. * or latest instead of a version range, or an unbounded version range using >/>=) is risky and almost certainly will break your package unexpectedly when a new major version of the dependency is released.

As with adding dependencies in general, updating them without any review carries security risks. In sensitive contexts you might want to consider gatekeeping every update by pinning the dependencies to their exact version.

Further reading:

Using lockfiles

Lockfiles are a snapshot of your full dependency tree, incl. all sub-dependencies, their precise versions and also integrity hashes. Having a lockfile allows npm to recreate the dependency tree exactly as it was. npm supports three lockfiles: package-lock.json and npm-shrinkwrap.json, and in npm 7+, yarn.lock, Yarn's format.

package-lock.json will get created automatically by default and it will not be published into the registry, i.e. it is only intended to be used for development purposes. npm recommends you commit this file into your source control.

You can choose to use the npm shrinkwrap command to instead create an npm-shrinkwrap.json file. It is the same in structure as the package-lock.json, but it will also be published together with your package in the registry, which means that not only people who develop the package, but also the users will receive the exact set of dependencies, and will be unable to dedupe transitive dependencies with the rest of their dependency graph.

Using lockfiles has downsides:

  • package-lock.json (and yarn.lock) is only used in development, so you may be testing your package with a different set of dependency versions than your users will be using it, which may occasionally lead to unexpected results.
  • no dependency updates will be received automatically with npm install.
  • using npm-shrinkwrap.json in a package intended to be used as a library may prevent deduplication of dependencies, increasing node_modules / bundle size.

You can disable lockfiles in your global or local .npmrc file by adding a package-lock=false line. This does however mean that npm will also ignore the npm-shrinkwrap.json files included in your dependencies.

Further reading:

Updating the dependencies manually

You can list outdated dependencies using the npm outdated command. If there are any in-range updates - you can use npm update command to install them.

You will need to modify your package.json version specifier to receive out-of-range updates.

There are several tools that will help you manage this from your local command line, such as npm-check and npm-check-updates, however you can also set up external tools to automatically create Pull Requests with updates.

Automatically keeping dependencies up to date

If your package has automated tests and CI/CD, you can use one of several services, free for Open Source Software, to automatically keep your dependencies up to date.

All of the below tools provide a GitHub App to create PRs with updates. They also support maintaining lock files to update sub-dependencies. Each tool also has built-in or community supported ways of automatically merging the PRs as long as tests pass, although this does carry some risk.

  • Dependabot
    • Supports other languages, besides JavaScript
  • Renovate
    • Supports other languages, besides JavaScript
    • Supports updating .travis.yml, Circle CI yml and .nvmrc, i.e. it can update the Node.js version you use for running tests
  • Snyk
    • Supports other languages, besides JavaScript
    • Supports ignoring specific deps, and limiting the amount of open simultaneous PRs to reduce noise level