Overview

I was chatting with someone earlier in the week who thought that managing Python virtual environments (venvs) manually would be onerous. I don’t do much Python development but it is something I reach for when shell or existing automation tools aren’t sufficient.

I’ve never found venvs that much of a hassle but thinking about his point of view made me think: “I bet you could make venvs fairly seamless with direnv…”. Then I figured: “Others are bound to have already done this…” And they have. cf: Further reading

Before jumping straight to pipenv & direnv some context and examples around the tools…

Background

Some basic definitions on the building blocks:

Most operating systems ship with a Python interpreter (often more than one) and the Python Standard Libraries pre-packaged (as RPMs, DEBs or whatever package format your distribution favours.)

There are hundreds of thousands of additional Python packages or modules in the ecosystem. Typically these are distributed via the community Python Package Index (PyPI). At time of writing this includes:

  • 606,397 projects
  • 6,561,707 releases
  • 13,265,297 files
  • 899,744 users

Many of the more popular ones may also be packaged and bundled with your distribution (Fedora, Debian, Arch (btw) etc.). Often, either a package you would like may not be maintained by your distribution vendor, or may not be repackaged as quickly as you would like, lagging behind the PyPY release by months or more.

pip

pip

is the package installer for Python. You can use pip to install packages from the Python Package Index and other indexes.

Running pip directly as a root user (without an additional wrapper or safety net) is NOT recommended and can break system Python installs and/or cause dependency hell.

Using pip as a normal, non-root user will typically install packages into ~/.local/. This can be moderately safe but if you have multiple Python projects with differing version requirements, this can still cause breakage at the user’s Python level. (The system Python should be safe so long as you do not pip install as root/via sudo.)

This illustrates what pip install-ing would look like if run as a normal user. --dry-run is used to model the ‘what if’ scenario without actually changing the target system.

[wmcdonald@fedora ~ ]$ pip install --dry-run boto
Defaulting to user installation because normal site-packages is not writeable
Collecting boto
  Using cached boto-2.49.0-py2.py3-none-any.whl.metadata (7.3 kB)
Using cached boto-2.49.0-py2.py3-none-any.whl (1.4 MB)
Would install boto-2.49.0

This illustrates what pip install-ing would look like if run for a package that is already installed, in this case by the system Python pre-packaged by the distribution vendor, Fedora.

[wmcdonald@fedora ~ ]$ pip install --dry-run boto3
Defaulting to user installation because normal site-packages is not writeable
Requirement already satisfied: boto3 in /usr/lib/python3.13/site-packages (1.36.7)
Requirement already satisfied: botocore<1.37.0,>=1.36.7 in /usr/lib/python3.13/site-packages (from boto3) (1.36.7)
Requirement already satisfied: jmespath<2.0.0,>=0.7.1 in /usr/lib/python3.13/site-packages (from boto3) (1.0.1)
Requirement already satisfied: s3transfer<0.12.0,>=0.11.0 in /usr/lib/python3.13/site-packages (from boto3) (0.11.2)
Requirement already satisfied: python-dateutil<3.0.0,>=2.1 in /usr/lib/python3.13/site-packages (from botocore<1.37.0,>=1.36.7->boto3) (2.8.2)
Requirement already satisfied: urllib3!=2.2.0,<3,>=1.25.4 in /usr/lib/python3.13/site-packages (from botocore<1.37.0,>=1.36.7->boto3) (1.26.20)
Requirement already satisfied: six>=1.5 in /usr/lib/python3.13/site-packages (from python-dateutil<3.0.0,>=2.1->botocore<1.37.0,>=1.36.7->boto3) (1.16.0)

[wmcdonald@fedora ~ ]$ rpm -qf /usr/lib/python3.13/site-packages/boto3/
python3-boto3-1.36.7-1.fc41.noarch

[wmcdonald@fedora ~ ]$ rpm -qi `rpm -qf /usr/lib/python3.13/site-packages/boto3/`
Name        : python3-boto3
Version     : 1.36.7
Release     : 1.fc41
Architecture: noarch
Install Date: Mon 03 Feb 2025 23:20:16 GMT
Group       : Unspecified
Size        : 2167282
License     : Apache-2.0
Signature   : RSA/SHA256, Tue 28 Jan 2025 18:02:25 GMT, Key ID d0622462e99d6ad1
Source RPM  : python-boto3-1.36.7-1.fc41.src.rpm
Build Date  : Tue 28 Jan 2025 15:35:17 GMT
Build Host  : buildvm-a64-32.iad2.fedoraproject.org
Packager    : Fedora Project
Vendor      : Fedora Project
URL         : https://github.com/boto/boto3
Bug URL     : https://bugz.fedoraproject.org/python-boto3
Summary     : The AWS SDK for Python
Description :
Boto3 is the Amazon Web Services (AWS) Software Development Kit (SDK) for
Python, which allows Python developers to write software that makes use of
services like Amazon S3 and Amazon EC2.

Beyond pip install, there are also:

  • download - download packages.
  • uninstall - uninstall packages.
  • freeze - output installed packages in requirements format.
  • inspect - inspect the python environment.
  • list - list installed packages.
  • show - show information about installed packages.

venv

venv

supports creating lightweight “virtual environments”, each with their own independent set of Python packages installed in their site directories.

Using the venv module, we can create a wrapper, something akin to a chroot or a container, into which we can safely pip install dependencies. The venv is typically activated / deactivated manually when required.

For example, first create a directory and switch into it:

[wmcdonald@fedora ~ ]$ mkdir pipdemo && cd $_
/home/wmcdonald/pipdemo

Run a pip install --dry-run as we illustrated previously to see that a package/module conflict exists with the system Python and its Boto3 module:

[wmcdonald@fedora pipdemo ]$ pip install --dry-run boto3
Defaulting to user installation because normal site-packages is not writeable
Requirement already satisfied: boto3 in /usr/lib/python3.13/site-packages (1.36.7)
<snip>

Check the packaged Boto3 version in the RPM package database. Run the default system Python, import the boto3 module and enumerate its current version:

[wmcdonald@fedora ~ ]$ rpm -qf /usr/lib/python3.13/site-packages/boto3/
python3-boto3-1.36.7-1.fc41.noarch

[wmcdonald@fedora pipdemo ]$ python
Python 3.13.1 (main, Dec  9 2024, 00:00:00) [GCC 14.2.1 20240912 (Red Hat 14.2.1-3)] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import boto3
>>> print(boto3.__version__)
1.36.7

Now, create a Python venv, python -m venv invokes the venv module, .venv is the name of the directory to store the virtual environment and is arbitrary, it’s common to use a ‘dotfile’ to hide this, and the directory name can reflect the purpose of the venv, or just be generic.

[wmcdonald@fedora pipdemo ]$ python -m venv .venv

And inspect its contents, note the activate script(s) in the bin directory:

[wmcdonald@fedora pipdemo ]$ tree -L 3 .venv/
.venv/
├── bin
│   ├── activate
│   ├── activate.csh
│   ├── activate.fish
│   ├── Activate.ps1
│   ├── pip
│   ├── pip3
│   ├── pip3.13
│   ├── python -> /usr/bin/python
│   ├── python3 -> python
│   └── python3.13 -> python
├── include
│   └── python3.13
├── lib
│   └── python3.13
│       └── site-packages
├── lib64 -> lib
└── pyvenv.cfg

source (or . ) the activate script to invoke or ‘enter’ the virtual environment:

Note: the prompt will change and prepend the venv name, to indicate the presence of an active venv.

[wmcdonald@fedora pipdemo ]$ . .venv/bin/activate
(.venv)[wmcdonald@fedora pipdemo ]$ 

Now again, run Python and attempt to import Boto3:

(.venv)[wmcdonald@fedora pipdemo ]$ python
Python 3.13.1 (main, Dec  9 2024, 00:00:00) [GCC 14.2.1 20240912 (Red Hat 14.2.1-3)] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import boto3
Traceback (most recent call last):
  File "<python-input-0>", line 1, in <module>
    import boto3
ModuleNotFoundError: No module named 'boto3'

We’ve confirmed that Boto3 “doesn’t exist” from the point-of-view of an active venv. Now --dry-run install Boto3 with pip.

Note: there are no package/module conflicts despite the system Boto3 still being installed/present, this is because it’s not present from the point-of-view of the active venv.

(.venv)[wmcdonald@fedora pipdemo ]$ pip install --dry-run boto3
Collecting boto3
  Downloading boto3-1.36.13-py3-none-any.whl.metadata (6.7 kB)
<snip>
>Would install boto3-1.36.13 botocore-1.36.13 jmespath-1.0.1 python-dateutil-2.9.0.post0 s3transfer-0.11.2 six-1.17.0 urllib3-2.3.0

Now, let’s install Boto3 and its dependencies inside the active venv:

(.venv)[wmcdonald@fedora pipdemo ]$ pip install boto3
Collecting boto3
  Using cached boto3-1.36.13-py3-none-any.whl.metadata (6.7 kB)
<snip>
Installing collected packages: urllib3, six, jmespath, python-dateutil, botocore, s3transfer, boto3
Successfully installed boto3-1.36.13 botocore-1.36.13 jmespath-1.0.1 python-dateutil-2.9.0.post0 s3transfer-0.11.2 six-1.17.0 urllib3-2.3.0

Start the Python interpreter, import the boto3 module (which is the verion we’ve just installed using pip, into our venv). Inspect the Boto3 __version__ and verify the module’s __file__:

Note #1: *it’s a later version of Boto3 than the packaged.

Note #2: *it’s the locally installed Boto3 from the venv.

(.venv)[wmcdonald@fedora pipdemo ]$ python
Python 3.13.1 (main, Dec  9 2024, 00:00:00) [GCC 14.2.1 20240912 (Red Hat 14.2.1-3)] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import boto3
>>> dir(boto3)
['DEFAULT_SESSION', 'NullHandler', 'Session', '__author__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__path__', '__spec__', '__version__', '_get_default_session', '_warn_deprecated_python', 'client', 'compat', 'docs', 'exceptions', 'logging', 'resource', 'resources', 'session', 'set_stream_logger', 'setup_default_session', 'utils']
>>> print(boto3.__version__)
1.36.13
>>> print(boto3.__file__)
/home/wmcdonald/pipdemo/.venv/lib64/python3.13/site-packages/boto3/__init__.py

Now we can deactivate the active venv using an alias which is created in the shell environment during the activate of the venv. And clean up our test directory:

(.venv)[wmcdonald@fedora pipdemo ]$ deactivate 
[wmcdonald@fedora pipdemo ]$ cd ..
[wmcdonald@fedora ~ ]$ rm -rf pipdemo/

pipenv

pipenv

is a Python virtualenv management tool that supports a multitude of systems and nicely bridges the gaps between pip, python (using system python, pyenv or asdf) and virtualenv.

pipenv smushes together pip and venv so you have a single tool that will manage the creation/enablement of venvs with the resolution and installation of dependencies.

pipenv recommend a --user installation and for Fedora, this is the simplest, safest route. pipenv is available pre-packaged as a DEB on both Debian and Ubuntu.

Once installed, create a project directory:

[vagrant@localhost ~]$ mkdir project-
[vagrant@localhost ~]$ ls -ld project-*
drwxr-xr-x. 1 vagrant vagrant 0 Feb  7 16:33 project-a

Verify that the pip-install-test module is not present in the default system Python:

[vagrant@localhost ~]$ python -c 'import pip_install_test'
Traceback (most recent call last):
  File "<string>", line 1, in <module>
    import pip_install_test
ModuleNotFoundError: No module named 'pip_install_test'

Switch into the test project directory:

[vagrant@localhost ~]$ cd project-a/
[vagrant@localhost project-a]$ 

Install a sample package into the first project directory (output truncated for legibility):

[vagrant@localhost project-a]$ pipenv install pip-install-test 
Creating a virtualenv for this project
Creating a Pipfile for this project...
Building requirements...
Resolving dependencies...
✔ Success!
To activate this project's virtualenv, run pipenv shell.
Alternatively, run a command inside the virtualenv with pipenv run.

Revalidate that the pip-install-test module is NOT available:

[vagrant@localhost project-a]$ python -c 'import pip_install_test'
Traceback (most recent call last):
  File "<string>", line 1, in <module>
    import pip_install_test
ModuleNotFoundError: No module named 'pip_install_test'

Now a) run with pipenv run and b) run through pipenv shell:

[vagrant@localhost project-a]$ pipenv run python -c 'import pip_install_test'
Good job!  You installed a pip module.

Now get back to work!

[vagrant@localhost project-a]$ pipenv shell
Launching subshell in virtual environment...
[vagrant@localhost project-a]$  source /home/vagrant/.local/share/virtualenvs/project-a-vnYESTNH/bin/activate
(project-a) [vagrant@localhost project-a]$ python -c 'import pip_install_test'
Good job!  You installed a pip module.

Now get back to work!
(project-a) [vagrant@localhost project-a]$ 

Exit the subshell:

(project-a) [vagrant@localhost project-a]$ exit
exit
[vagrant@localhost project-a]$ 

And now clean up the pipenv-created venv and the Pipfile and its lockfile.

[vagrant@localhost project-a]$ pipenv --rm && rm Pipfile*
Removing virtualenv (/home/vagrant/.local/share/virtualenvs/project-a-vnYESTNH)...
[vagrant@localhost project-a]$ 

direnv

direnv

is an extension for your shell. It augments existing shells with a new feature that can load and unload environment variables depending on the current directory.

Once installed, and hooked into your shell direnv can be used to trigger specific behaviours when you switch into a specific directory. For example, if you switch into a directory that has a venv you could automatically activate the venv on entry, and deactivate if you switch out.

See the direnv quick demo for a… quick demo.

How-to

Now we have an understanding of all the component parts, we can use direnv to automatically load/unload a pipenv on entry into the directory “containing” the pipenv-managed venv and installed modules .

First create the pipenv

[vagrant@localhost ~]$ mkdir project-a && cd $_
[vagrant@localhost project-a]$ pipenv install pip-install-test 

Configure and enable direnv to ‘layout’ the pipenv and run pipenv graph to enumerate installed Python module(s):

[vagrant@localhost project-a]$ echo 'layout pipenv' > .envrc
[vagrant@localhost project-a]$ direnv allow
[vagrant@localhost project-a]$ pipenv graph
pip-install-test==0.5

Switch out of the pipenv directory, observe that direnv unloads.

[vagrant@localhost project-a]$ cd ~
direnv: unloading

Run a test Python module import from the user ${HOME}, verify that the module cannot be found:

[vagrant@localhost ~]$ python -c 'import pip_install_test'
Traceback (most recent call last):
  File "<string>", line 1, in <module>
    import pip_install_test
ModuleNotFoundError: No module named 'pip_install_test'

Switch back into the directory, observe that direnv activates the pipenv, re-run the test Python module import and verify that the test module is available:

[vagrant@localhost ~]$ cd project-a/
direnv: loading ~/project-a/.envrc
direnv: export +PIPENV_ACTIVE +VIRTUAL_ENV ~PATH
[vagrant@localhost project-a]$ python -c 'import pip_install_test'
Good job!  You installed a pip module.

Now get back to work!

Other tools

There’s a broad range of Python package and dependency management tooling in the ecosystem. These include:

  • Tox - useful for managing multiple combinatorial states of versions and modules.
  • uv - reputedly incredibly fast and efficient dependency management written in Rust.

Further reading

Notes

If you try to use layout python or layout python3 instead of layout pipenv this will trigger an exception with direnv =< 2.34. This applies in Fedora 41, Debian 12 & Ubuntu 24.04. Bug in-progress: https://bugzilla.redhat.com/show_bug.cgi?id=2344401