uvlink is a Python CLI tool that caches virtual environments outside your project and symlinks them back. Perfect for uv users who sync code to Dropbox, Google Drive, or iCloud. Only your source code syncs, not gigabytes of .venv dependencies.
Caution
Since v0.6.0, the cache directory includes the venv type in its folder name and stores the environment under a matching subdirectory (e.g. ~/.local/share/uvlink/cache/<project-name>-<hash>-.venv/.venv). Delete caches created with older versions and rerun uvlink link to migrate.
- Python 3.12+
- macOS, Linux, and Windows(beta support)
Note
Windows support is considered beta as of v0.8.0. It uses directory junctions as a fallback when symlinks are not available, which does not require admin privileges or Developer Mode. Please report any issues you encounter on Windows!
$ uv tool install uvlinkThis installs the CLI into your ~/.local/bin (or platform equivalent) with isolated dependencies handled by Astral's uv.
$ pip install uvlinkNavigate to any Python project (which may be created by uv init for instance) and run uvlink link:
$ cd /path/to/your/project
$ uvlink linkThe link command creates a .venv symlink in your project pointing to a cached environment. By default, it uses .venv as the symlink name because uv sync installs into .venv by default. You can use a different name if needed, for example:
$ uvlink link myenvThis creates a myenv symlink instead.
The cached environment is stored under $XDG_DATA_HOME/uvlink/cache/<project-name>-<hash>-<venv_name>/<venv_name> (or ~/.local/share/uvlink/cache/... if XDG_DATA_HOME is not set).
After linking, you can use the virtual environment just like you would with uv whithin the project directory:
$ source .venv/bin/activate # Activate the environment
$ uv sync # Install dependencies into the cached .venv
$ uv run python script.py # Run commands in the environmentSince .venv exists as a symlink in your project directory, all standard activation methods work. Your cloud service will ignore the heavy virtual environment files since they're stored in the cache.
List all cached environments:
$ uvlink lsShows all projects with cached environments and their link status. Projects where the symlink has been removed (e.g., via rm .venv) will show as "Not Linked" (❌). You can relink them by running uvlink link again in that project directory.
Clean up unlinked caches:
$ uvlink gcRemoves cached environments for projects that no longer have working symlinks (marked as "Not Linked" in uvlink ls), freeing up disk space.
uvlink ships with a Typer CLI. Run uvlink --help for all available options and commands.
Specify a project directory:
$ uvlink --project-dir /path/to/project linkWorks from any location without needing to cd into the project directory first.
Swap the .venv folder name:
$ uvlink --project-dir /path/to/project link [VENV_TYPE]Custom [VENV_TYPE] are helpful when sharing a cache across tooling expectations (e.g., .venv-prod, .venv-dev).
Custom Cache Location:
The default cache location is $XDG_DATA_HOME/uvlink/cache if XDG_DATA_HOME is set, otherwise it falls back to ~/.local/share/uvlink/cache.
To override the cache location, you can either:
- Set the
XDG_DATA_HOMEenvironment variable:$ export XDG_DATA_HOME=/my/custom/path $ uvlink link
Warning
This will affect other software using XDG_DATA_HOME environment variable as well!
-
Use the
--cache-rootoption (must be used consistently across all commands):$ uvlink --cache-root /path/to/cache link $ uvlink --cache-root /path/to/cache ls $ uvlink --cache-root /path/to/cache gc
Note: When using
--cache-root, you must specify it for every command (link,ls,gc). Consider setting up a shell alias to avoid repetition:$ alias uvlink='uvlink --cache-root /my/cache/root'
uvlink stores environments under $XDG_DATA_HOME/uvlink/cache/<project-name>-<hash>-<venv_name>/<venv_name> (or ~/.local/share/uvlink/cache/... if XDG_DATA_HOME is not set) and creates a symlink <venv_name> in your project directory pointing to that cached location.
Each project receives a stable hash based on its absolute path, so repeated runs target the same cache location. The symlink behaves like a regular .venv directory for most purposes. You can activate it with source .venv/bin/activate or use uv run commands as usual.
What uvlink does:
- Creates a symlink
.venv(or your custom name) in your project directory - Stores the actual virtual environment in a centralized cache location
What uvlink does NOT do:
- It does not install packages (use
uv sync,uv add,uv pip install, etc.) - It does not remove symlinks (use
rm .venvif you want to unlink and runuvlink gcafter unlink to clean up the unlinked caches.) - It does not manage or activate environments (use standard
uvcommands)
Issues and pull requests are welcome. Please keep docstrings and comments in the Google style already used throughout the codebase and run the included linters/formatters before submitting. A pre-commit configuration is provided that runs Ruff; install it with pre-commit install to match formatting.
Also, feel free to make comments/discussion at
