Skip to content

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:

arxiv

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

RUN pip install -r requirements.txt

to

RUN uv pip install -r requirements.txt

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.

RUN uv pip install -r requirements.txt

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

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