uv - An extremely fast Python package and project manager, written in Rust¶
There exist a broad selection of package managers that manage virtual environments and packages for us. Some of the most popular ones are conda, pip and virtualenv or virtualenvwrapper. They can almost all do the same things:
- Install packages
- Resolve dependencies
- Managing package versions
- Managing virtual environments
Why uv?¶
You might ask yourself the question of why do we need yet another tool for this? Because it is faster and faster is better. How fast you ask? Between 10 and 100x faster, depending on the task. Besides that it is also fun to use and support something new. But before we get started with some examples, let's go over the key design decisions that make uv so fast.
Design Decisions¶
- uv comes as a standalone binary and was written in Rust
- uv uses a global module cache, that prevents re-downloading of packages when you set up a new project on the same machine
- During download packages are written directly to disk without memory overhead (Copy-on-Write)
- uv is a drop-in replacement for pip, pip-tools and can manage virtual environments like virtualenv and conda.
Next, let's go through the most important commands and at the end we'll upgrade the Arxiv Publications Tracker to use uv instead of pip.
The most important commands¶
Command | Description | Example | Parameters/Notes |
---|---|---|---|
uv pip | Drop-in replacement for pip commands | uv pip install requests | Accepts all standard pip arguments and flags |
uv pip compile | Resolves dependencies and creates requirements.txt | uv pip compile requirements.in | requirements.in contains direct dependencies without versions (e.g., "requests\nflask"). The output requirements.txt will contain all dependencies with pinned versions |
uv pip sync | Installs packages from requirements.txt | uv pip sync requirements.txt | Ensures exact versions from requirements.txt are installed. Will remove packages not in requirements.txt |
uv venv | Creates a new virtual environment | uv venv | Creates in current directory as .venv by default. Use --path to specify different location |
uv sync | Syncs the dependencies to the virtual environment | uv sync | Reads from requirements.txt or pyproject.toml . Use --python to specify Python version |
uv add | Adds a package to the dependencies | uv add requests | Can specify version constraints (e.g., requests>=2.28.0 ). Updates requirements files automatically |
uv remove | Removes a package from the dependencies | uv remove requests | Removes both the package and its unused dependencies |
uv list | Lists all installed packages | uv list | Use --freeze to output in requirements.txt format |
uv run | Runs a Python script in an isolated environment | uv run script.py | Creates temporary venv with dependencies from requirements.txt or pyproject.toml . Use --python to specify version |
About pyproject.toml¶
The pyproject.toml
file is a standardized configuration file for Python projects (defined in PEP 518). When using uv, this file can specify your project's dependencies instead of using requirements.txt. Here's an example:
[project]
name = "my-project"
version = "1.0.0"
dependencies = [
"requests>=2.28.0",
"flask~=2.0.0",
"pandas>=2.0.0"
]
[project.optional-dependencies]
dev = [
"pytest>=7.0.0",
"black>=23.0.0"
]
The advantages of using pyproject.toml include:
- Single source of truth for project metadata and dependencies
- Support for optional dependency groups (like development tools)
- Better integration with modern Python packaging tools
- Ability to specify build requirements and project metadata
When using uv commands like uv sync
or uv run
, it will automatically detect and use dependencies specified in either pyproject.toml or requirements.txt, with pyproject.toml taking precedence if both exist.
In our demo we'll skip creating a pyproject.toml file and only use requirements.txt.
Demo¶
The Arxiv Publications Tracker is a very simple tool you can use to find new papers on arXiv.org matching your search criteria. I use it to find new papers on LLM driven AI Agents.
The tool runs inside a Docker container, and it uses pip to fetch python dependencies.
Our original Dockerfile
looks like this:
# Use a minimal Python 3.11 image
FROM python:3.11-slim
# Set the working directory
WORKDIR /app
# Copy the script into the container
COPY arxiv_tracker.py /app/arxiv_tracker.py
COPY requirements.txt /app/requirements.txt
# Install dependencies
RUN pip install -r requirements.txt
# Run the script as the entrypoint
ENTRYPOINT ["python", "/app/arxiv_tracker.py"]
Our requirements.txt
file contains the dependency to the arxiv python package:
To upgrade this project to uv we'll need to install uv in the docker container and then modify the RUN pip install -r requirements.txt
command.
Since we know uv pip is a drop-in replacement for pip we simply change the command
to
If you want to install uv on your machine follow the installation guide for uv on PyPi or Astral.sh.
If we run the build script ./build_and_run.sh
we get the following error:
=> ERROR [arxiv_watchdog 5/5] RUN uv pip install -r requirements.txt
0.250 /bin/sh: 1: uv: not found
It seems that uv is not part of our docker base image python:3.11-slim
and therefore not found when we try to run it. This can easily be fixed, because uv can be installed using pip, and we know we already had pip available in the container.
We simply have to add the following line to our Dockerfile, before we can use uv.
If we run the build script again, we see a different error:
=> ERROR [arxiv_watchdog 6/6] RUN uv pip install -r requirements.
.351 error: No virtual environment found; run `uv venv` to create an environment, or pass `--system` to install into a non-virtual environment
This is because uv requires us to create a virtual environment or explicitly specify --system
if we don't want to use one. Since we have nothing else running in the docker container, we'll be fine using --system.
The updated and final version of our Dockerfile looks like this:
# Use a minimal Python 3.11 image
FROM python:3.11-slim
# Set the working directory
WORKDIR /app
# Copy the script into the container
COPY arxiv_tracker.py /app/arxiv_tracker.py
COPY requirements.txt /app/requirements.txt
# Install dependencies
RUN pip install uv
RUN uv pip install --system -r requirements.txt
# Run the script as the entrypoint
ENTRYPOINT ["uv", "run", "/app/arxiv_tracker.py"]
Notice that we have modified the ENTRYPOINT
to run the arxiv_tracker.py
script using uv
instead of python
directly.
Conclusion¶
This is it! We have successfully upgraded the arXiv publications tracker to use uv instead of pip. If you are interested, you can get the updated version of the arXiv publications tracker here and give it a try. To learn more about what you can do with uv, check out the official uv documentation.
Takeaways
- uv is ultra fast
- uv is a drop in replacement for pip and pip-tools
- uv can replace conda and virtualenv
- read the official uv documentation to learn more
Links and Resources¶
- uv: Python packaging in Rust by the creator of uv Charlie Marsh
- uv on Astral.sh - The official documentation for uv
- uv on PyPi - The mirrored documentation for uv on PyPi
- Arxiv Publications Tracker - A Python based tool to incrementally find the latest publications on arXiv
Other Package Managers
- conda - OS-agnostic, system-level binary package and environment manager
- pip - pip is the package installer for Python
- pip-tools - A set of command line tools to help you keep your pip-based packages fresh
- virtualenv - a tool to create isolated Python environments
- virtualenvwrapper - a tool to manage virtual environments