Skip to content

Rush to pnpm Migration Guide

Matthew Podwysocki edited this page Apr 1, 2025 · 5 revisions

The Azure SDK for JavaScript is migrating from Rush to pnpm as part of the Azure SDK for JavaScript modernization efforts. This is a guide to show how to migrate from Rush commands to pnpm commands for common scenarios.

Scenarios

These are the following scenarios covered:

Installation

Rush

Commonly Rush was installed using npm and installing globally such as the following:

npm install -g @microsoft/rush

Once installed, we then initialize Rush with:

rush update

pnpm

pnpm can be installed as noted in the installation guide either through a shell script, package manager such as WinGet or Homebrew, or npm with or without Corepack.

Once installed, you should run install inside the SDK project.

pnpm install

Common Commands

All common commands from Rush are directly translated to pnpm directly for the following, by simply replacing rush with pnpm

  • build: Builds the project
  • build:test: Builds the project for testing
  • build:samples: Builds the samples
  • clean: Cleans the workspace and project
  • check-format: Checks the formatting using Prettier
  • format: Formats the project using Prettier
  • integration-test: Executes the integration tests for both browser and Node.js
  • integration-test:browser: Executes the integration tests for the browser
  • integration-test:node: Excutes the integration tests for Node.js
  • lint: Checks the linting using ESLint
  • lint:fix: Checks the linting using ESLint and fixes the issues if they are fixable
  • test: Executes all tests for both unit and integration tests
  • unit-test: Executes the unit tests for both the browser and Node.js
  • unit-test:browser: Executes the unit tests for the browser
  • unit-test:node: Excutes the unit tests for Node.js
  • update-snippets: Updates the snippets for documentation and code

Using Commands

Rush

With Rush, we could run against the entire repository such as building everything in the workspace using:

rush build

For building a single project, uses the -t option.

rush build -t <project-name>

Accepts multiple projects to build at once with usage of multiple -t

rush build -t <project-name-1> -t <project-name-2>

When inside a service SDK folder, we can run the local commands using rushx such as:

rushx build

pnpm

To target everything we can do the same with pnpm, for example to build the entire workspace.

pnpm build

To build a single project, we can use the --filter=<package-name>... option. Note in order to build the project and its dependencies, the ... is required after the project name, however, if you just want to build the project without its depdendencies, you can leave off the ....

pnpm build --filter=<project-name>...

Multiple filters can be used to build multiple projects such as the following:

pnpm build --filter=<project-name-1>... --filter<project-name-2>...

When inside a project, we can build the project by itself with depdendencies such as the following using the -r for recursive:

pnpm -r build

If we just want to build the project by itself and not its dependencies, we can omit the -r option:

pnpm build

Workspace Management

Both Rush and pnpm support workspaces, so let's go over the differences.

Rush

With Rush, you had the rush.json file at the root where we had our workspace projects listed in the projects section of the JSON, specifying the location, the package name and versioning policy. This also kept the stored information about the workspace in the common/config/rush folder with the lock file. All packages had to be explicitly added to the workspace through the rush.json and then rush update called to include the package.

pnpm

Using pnpm, we can manage the workspace with the pnpm-workspace.yaml file at the root of the repository. By default, when adding a new package to the repository under the sdk folder, it should be picked up immediately once pnpm install has been called. If the package has not been picked up, it may have been excluded by one of our ignore patterns. Please work with the team if that is the case.

Inside the pnpm-workspace.yaml file, we have several sections:

  • packages (our workspace packages)
  • catalog (default catalog)
  • catalogs (our named catalogs)

For the packages section, this should include all we plan to build and ship as part of the SDK with both things in common/tools as well as the sdk folder. By default, we should not need to edit this section to add any new SDKs to the workspace. In order to exclude something from being part of the workspace, ensure the ! syntax is used for example !sdk/core/should-not-be-added.

In the catalog section, using the Catalog Protocol, we keep our packages such as TypeScript, TSLib and other package references used across the entire set of packages. This should be kept rather small and instead, we should focus on named catalogs for specific focus areas to version things together.

Named Catalogs are a great feature of pnpm and are often used in our pnpm-workspace.yaml file. This allows us to separate out different versions of libraries such as @azure-tools/test-recorder V3 and V4, for example. We also have sections for OTEL, identity, testing and other focus areas to allow them to be easily versioned. Note that we have an internal catalog, so we can break any circular references, for example, not needing to bring in @azure/identity to all packages as a workspace causing it to be built, and instead just as a normal reference. 

Currently, we have the following named catalogs:

  • testing: vitest and modern testing
  • testinglegacy: Legacy testing needs
  • corelroV2: for packages that use the older @azure/core-lro version 2
  • credentialV1: For packages that use @azure-tools/test-credential version 1
  • recorderV3: For packages that use @azure-tools/test-recorder version 3
  • internal: internal packages used as to prevent circular references
  • identity: MSAL and other identity packages
  • amqp: AMQP related packages
  • polyfills: Polyfills necessary for Node.js and the browser
  • otel: All OpenTelemetry libraries.

Package Management

Adding packages should be able to do be done via both the command line and editing the package.json of the project individually.

Rush

In Rush, we can add a package to a project by using the CLI:

# Assume latest version
rush add --package <package-name>

# Add a specific version with the ~ version
rush add --package <package-name>@<version>

# Add a specific version with a ^ version
rush add --package <package-name>@<version> --caret

# Add to devDependencies
rush add --package <package-name>@<version> --dev

In addition, we can edit the package.json with the correct version and then call rush update.

pnpm

With pnpm, we strive to use workspace and catalog references as much as possible. This allows us to be consistent in our versioning to ensure that we are up to date with our latest depdendencies.

With pnpm, we can add packages with similar examples to Rush above:

# Assume latest version
pnpm add <package-name>

# Add a specific version with the ~ version
pnpm add <package-name>#semver:~<version> 

# Add a specific version with a ^ version
rush add <package-name>#semver:^<version> 

# Add to devDependencies
pnpm add <package-name>@<version> -D

In addition, we can edit the package.json with the correct version and then call pnpm install. Note that we prefer to use catalog and workspace references as applicable. Reference the pnpm-workspace.yaml file for the default catalog and named catalogs for the references you may need.

Here is an example such as with the @azure/ai-projects package:

  "dependencies": {
    "@azure-rest/core-client": "workspace:*",
    "@azure/abort-controller": "workspace:*",
    "@azure/core-auth": "workspace:*",
    "@azure/core-rest-pipeline": "workspace:*",
    "@azure/core-util": "workspace:*",
    "@azure/logger": "workspace:*",
    "@azure/core-lro": "workspace:*",
    "tslib": "catalog:",
    "@azure/core-paging": "workspace:*",
    "@azure/core-sse": "workspace:*",
    "@azure/core-tracing": "workspace:*"
  },
  "devDependencies": {
    "@azure/dev-tool": "workspace:*",
    "@azure/eslint-plugin-azure-sdk": "workspace:*",
    "@azure/identity": "catalog:internal",
    "@azure/opentelemetry-instrumentation-azure-sdk": "catalog:otel",
    "@azure/monitor-opentelemetry-exporter": "catalog:otel",
    "@azure-tools/test-credential": "workspace:*",
    "@azure-tools/test-recorder": "workspace:*",
    "@azure-tools/test-utils-vitest": "workspace:*",
    "@opentelemetry/api": "catalog:otel",
    "@opentelemetry/instrumentation": "catalog:otel",
    "@opentelemetry/sdk-trace-node": "catalog:otel",
    "@vitest/browser": "catalog:testing",
    "@vitest/coverage-istanbul": "catalog:testing",
    "@types/node": "catalog:",
    "dotenv": "catalog:testing",
    "eslint": "catalog:",
    "playwright": "catalog:testing",
    "typescript": "catalog:",
    "vitest": "catalog:testing"
  },