.agents/skills/pnpm/references/best-practices-migration.md
Guide for migrating existing projects from npm or Yarn to pnpm.
# Remove npm lockfile and node_modules
rm -rf node_modules package-lock.json
# Install with pnpm
pnpm install
# Remove yarn lockfile and node_modules
rm -rf node_modules yarn.lock
# Install with pnpm
pnpm install
pnpm can import existing lockfiles:
# Import from npm or yarn lockfile
pnpm import
# This creates pnpm-lock.yaml from:
# - package-lock.json (npm)
# - yarn.lock (yarn)
# - npm-shrinkwrap.json (npm)
pnpm is strict about dependencies. If code imports a package not in package.json, it will fail.
Problem:
// Works with npm (hoisted), fails with pnpm
import lodash from 'lodash' // Not in dependencies, installed by another package
Solution: Add missing dependencies explicitly:
pnpm add lodash
pnpm reports peer dependency issues by default.
Option 1: Let pnpm auto-install:
# .npmrc (default in pnpm v8+)
auto-install-peers=true
Option 2: Install manually:
pnpm add react react-dom
Option 3: Suppress warnings if acceptable:
{
"pnpm": {
"peerDependencyRules": {
"ignoreMissing": ["react"]
}
}
}
Some tools don't work with symlinks. Use hoisted mode:
# .npmrc
node-linker=hoisted
Or hoist specific packages:
public-hoist-pattern[]=*eslint*
public-hoist-pattern[]=*babel*
If native modules fail, try:
# Rebuild all native modules
pnpm rebuild
# Or reinstall
rm -rf node_modules
pnpm install
Create pnpm-workspace.yaml:
packages:
- 'packages/*'
Update internal dependencies to use workspace protocol:
{
"dependencies": {
"@myorg/utils": "workspace:^"
}
}
Install:
rm -rf node_modules packages/*/node_modules package-lock.json
pnpm install
Remove Yarn-specific files:
rm yarn.lock .yarnrc.yml
rm -rf .yarn
Create pnpm-workspace.yaml matching workspaces in package.json:
packages:
- 'packages/*'
Update package.json - remove Yarn workspace config if not needed:
{
// Remove "workspaces" field (optional, pnpm uses pnpm-workspace.yaml)
}
Convert workspace references:
// From Yarn
"@myorg/utils": "*"
// To pnpm
"@myorg/utils": "workspace:*"
pnpm can replace Lerna for most use cases:
# Lerna: run script in all packages
lerna run build
# pnpm equivalent
pnpm -r run build
# Lerna: run in specific package
lerna run build --scope=@myorg/app
# pnpm equivalent
pnpm --filter @myorg/app run build
# Lerna: publish
lerna publish
# pnpm: use changesets instead
pnpm add -Dw @changesets/cli
pnpm changeset
pnpm changeset version
pnpm publish -r
Most npm/Yarn settings work in pnpm's .npmrc:
# Registry settings (same as npm)
registry=https://registry.npmjs.org/
@myorg:registry=https://npm.myorg.com/
# Auth tokens (same as npm)
//registry.npmjs.org/:_authToken=${NPM_TOKEN}
# pnpm-specific additions
auto-install-peers=true
strict-peer-dependencies=false
Most scripts work unchanged. Update pnpm-specific patterns:
{
"scripts": {
// npm: recursive scripts
"build:all": "npm run build --workspaces",
// pnpm: use -r flag
"build:all": "pnpm -r run build",
// npm: run in specific workspace
"dev:app": "npm run dev -w packages/app",
// pnpm: use --filter
"dev:app": "pnpm --filter @myorg/app run dev"
}
}
Update CI configuration:
# Before (npm)
- run: npm ci
# After (pnpm)
- uses: pnpm/action-setup@v4
- run: pnpm install --frozen-lockfile
Add to package.json for Corepack:
{
"packageManager": "[email protected]"
}
For large projects, migrate gradually:
pnpm import to create lockfileIf migration causes issues:
# Remove pnpm files
rm -rf node_modules pnpm-lock.yaml pnpm-workspace.yaml
# Restore npm
npm install
# Or restore Yarn
yarn install
Keep old lockfile in git history for easy rollback.
<!-- Source references: - https://pnpm.io/installation - https://pnpm.io/cli/import - https://pnpm.io/limitations -->