Automatically switching Node.js version per project with nvm and a .nvmrc file
When working on multiple Node.js projects, you often need to switch between different Node.js versions. One project might run on Node.js 16, while another required Node.js 18. Manually switching versions every time you switch projects folders can be annoying and error-prone.
Luckily, there is a simple solution to this problem. By using nvm, you can easily switch between different Node.js versions. And by adding a .nvmrc
file to your project, you can automatically switch to the correct Node.js version when you enter the project folder.
Here’s how you can set up automatic Node.js version switching per project with nvm
and a .nvmrc
file:
Installing nvm
First, you need to install nvm on your system. You can find the installation instructions on the nvm GitHub page. Follow the instructions for your operating system to install nvm.
curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.40.1/install.sh | bash
Then, restart your terminal and verify that nvm
is installed:
nvm --version
Creating a .nvmrc
file
Next up, you need to create a .nvmrc
file in your project folder. This file will contain the Node.js version that your project requires. For example, if your project requires Node.js version 18, create a .nvmrc
file with the following content:
18
This will tell nvm
to use Node.js version 18 when you enter the project folder.
to manually switch to the version specified in the .nvmrc
file, you can run the following command in your project folder:
nvm use
Automatically switching Node.js version with a shell script
To automate this process, we can add a script in our shell configuration file. This script detects when a .nvmrc
file is present in a folder and automatically switches to the Node.js version specified in the file.
For Bash (.bashrc
or .bash_profile
)
Add the following script to your .bashrc
or .bash_profile
file:
cdnvm() {
command cd "$@" || return $?
nvm_path="$(nvm_find_up .nvmrc | command tr -d '\n')"
# If there are no .nvmrc file, use the default nvm version
if [[ ! $nvm_path = *[^[:space:]]* ]]; then
declare default_version
default_version="$(nvm version default)"
# If there is no default version, set it to `node`
# This will use the latest version on your machine
if [ $default_version = 'N/A' ]; then
nvm alias default node
default_version=$(nvm version default)
fi
# If the current version is not the default version, set it to use the default version
if [ "$(nvm current)" != "${default_version}" ]; then
nvm use default
fi
elif [[ -s "${nvm_path}/.nvmrc" && -r "${nvm_path}/.nvmrc" ]]; then
declare nvm_version
nvm_version=$(<"${nvm_path}"/.nvmrc)
declare locally_resolved_nvm_version
# `nvm ls` will check all locally-available versions
# If there are multiple matching versions, take the latest one
# Remove the `->` and `*` characters and spaces
# `locally_resolved_nvm_version` will be `N/A` if no local versions are found
locally_resolved_nvm_version=$(nvm ls --no-colors "${nvm_version}" | command tail -1 | command tr -d '\->*' | command tr -d '[:space:]')
# If it is not already installed, install it
# `nvm install` will implicitly use the newly-installed version
if [ "${locally_resolved_nvm_version}" = 'N/A' ]; then
nvm install "${nvm_version}";
elif [ "$(nvm current)" != "${locally_resolved_nvm_version}" ]; then
nvm use "${nvm_version}";
fi
fi
}
alias cd='cdnvm'
cdnvm "$PWD" || exit
Restart your terminal or run source ~/.bashrc
to apply the changes.
For Zsh (.zshrc
)
Add the following script to your .zshrc
file:
# place this after nvm initialization!
autoload -U add-zsh-hook
load-nvmrc() {
local nvmrc_path
nvmrc_path="$(nvm_find_nvmrc)"
if [ -n "$nvmrc_path" ]; then
local nvmrc_node_version
nvmrc_node_version=$(nvm version "$(cat "${nvmrc_path}")")
if [ "$nvmrc_node_version" = "N/A" ]; then
nvm install
elif [ "$nvmrc_node_version" != "$(nvm version)" ]; then
nvm use
fi
elif [ -n "$(PWD=$OLDPWD nvm_find_nvmrc)" ] && [ "$(nvm version)" != "$(nvm version default)" ]; then
echo "Reverting to nvm default version"
nvm use default
fi
}
add-zsh-hook chpwd load-nvmrc
load-nvmrc
Then restart your terminal or run source ~/.zshrc
to apply the changes.
For Fish (.config/fish/config.fish
)
This requires you th have bass installed.
And some files to be created in the ~/.config/fish/functions
folder.
~/.config/fish/functions/nvm.fish
:
function nvm
bass source ~/.nvm/nvm.sh --no-use ';' nvm $argv
end
~/.config/fish/functions/nvm_find_nvmrc.fish
:
function nvm_find_nvmrc
bass source ~/.nvm/nvm.sh --no-use ';' nvm_find_nvmrc
end
~/.config/fish/functions/load_nvm.fish
:
function load_nvm --on-variable="PWD"
set -l default_node_version (nvm version default)
set -l node_version (nvm version)
set -l nvmrc_path (nvm_find_nvmrc)
if test -n "$nvmrc_path"
set -l nvmrc_node_version (nvm version (cat $nvmrc_path))
if test "$nvmrc_node_version" = "N/A"
nvm install (cat $nvmrc_path)
else if test "$nvmrc_node_version" != "$node_version"
nvm use $nvmrc_node_version
end
else if test "$node_version" != "$default_node_version"
echo "Reverting to default Node version"
nvm use default
end
end
And then in your ~/.config/fish/config.fish
:
# You must call it on initialization or listening to directory switching won't work
load_nvm > /dev/stderr
Restart your terminal or run source ~/.config/fish/config.fish
to apply the changes.
Now, whenever you cd
into a project with a .nvmrc
file, the shell will automatically switch to the correct Node.js version. This setup makes it easy to work on multiple Node.js projects with different versions without having to worry about switching versions manually.
last updated: 2025-03-11
I hope this post was helpful to you. If you have any questions or feedback, feel free to contact me on Twitter/X, Bluesky, or Mastodon. I’d love to hear from you! 🚀