Virtual Code Environments#

In this chapter, you’re going to learn about virtual coding environments. These allow you to isolate all of the packages that you’re using to do analysis for one project from the set of packages you might need for a different project. They’re an important part of creating a reproducible analytical pipeline (more on that in Reproducible Analysis but a key benefit is that others can reproduce the environment you used) and it’s best practice to have an isolated environment per project.

If you’re going through this book using uv, you are already using virtual environments even though you may not have realised it! uv creates a different Python installation (with its attendant packages) for each project folder you use.

It may be easier to illustrate what separate environments are for with an example. Let’s say you’re using Python 3.8, statsmodels, and pandas for one project, project A. And, for project B, you’re using Python 3.9 with numpy and scikit-learn. Even with the same version of Python, best practice would be to have two separate virtual Python environments: environment A, with everything needed for project A, and environment B, with everything needed for project B. For the case where you’re using different versions of Python, this isn’t just best practice, it’s essential.

In Python, there are multiple tools for managing different environments. Of those, the easiest to work with is uv as it handles Python, packages, and reproducibility. However, if you’re using complex packages such as pymc you may prefer to use Miniconda (conda for short) instead as this offers pre-compiled code.

If you’re just getting going with coding, this book recommends that you use uv.

Using uv to Manage Environments#

uv automatically creates a virtual environment per project folder so that each project you do gets its own distinct and independent Python installation. As well as providing virtual environments by default, it can:

  • create virtual environments on a per project basis

  • manage Python package dependencies, and logs them automatically

  • manage Python package versions, and logs them automatically

  • manager Python versions (ie the language itself)

  • install all needed dependencies and packages from previously auto-generated (by uv) lists of versions

  • (advanced) build Python packages

  • (advanced) upload Python packages to PyPI where they can be installed (ie make your project pip installable by anyone in the world)

The environment that uv creates is tied to your project folder. After navigating to your project folder on the command line (more on the command line in The Command Line), use

uv init

This will create a (human-readable) pyproject.toml file that lists the packages needed for your project. It will also create a virtual Python environment just for your project.

To add and install packages, the command is uv add package-name. As you add packages, you will see that the pyproject.toml file automatically becomes populated with what you installed. Here’s an excerpt of a typical pyproject.toml file designed for reproducibility:

requires-python = ">=3.9"
dependencies = [
    "click>=8.1.7",
    "ipykernel>=6.29.5",
    "numpy>=2.0.2",
    "pandas-stubs>=2.2.2.240807",
    "pandas>=2.2.3",
    "polars>=1.17.1",
    "pyarrow>=18.1.0",
    "pygments>=2.18.0",
    "rich>=13.9.4",
    "typeguard>=4.4.1",
]

The first line here says that this runs on any Python version greater than 3.9. The rest sets out the main dependencies, or packages directly needed for the project, and their minimum versions.

A second file, uv.lock, is also created. This “locks” down the dependencies of your packages. It’s also human-readable, just not very compelling. Visual Studio Code will happily open both types of file. In that file is a list of every package directly used, the version that’s installed, and the dependencies of that package and what versions of those is installed.

Using uv Environments#

To use the virtual environment to run scripts or tools installed using uv add, you will need to use uv run command-name rather than the usual command-name. For example, if you have a script analysis.py, you would call it with uv run python analysis.py.

Visual Studio Code makes this easier though: just as with conda environments, you can set your interactive window or jupyter notebooks to execute with a specific uv environment. uv environments are by default found in your project folder under the .venv directory.

uv is especially strong for reproducibility. Imagine you wish to have a co-author or colleague install everything they need for the project. If you send them the automatically generated pyproject.toml and uv.lock files then all they need is to run uv sync --frozen. This will install all of (exactly the same) packages needed to run the code!

Using Miniconda to Manage Python Environments#

Much of these two subsections is covered by the Miniconda documentation on managing virtual environments.

Creating Environments#

If you’re using Miniconda for Python, you manage and change environments on the command line (more on the command line in The Command Line) too. Before following these instructions, check that you have Miniconda installed and activated. You should see something like (base) username@computername:~$ on the command line (base is the default conda environment).

To create a new environment called “myenv” with a specific version of Python (but no extra packages installed), it’s

conda create -n myenv python=3.8

where you can of course specify other versions of Python by changing the number. To throw in a package or two, just add them to the end, for example

conda create -n myenv python=3.8 pandas jupyter

You can see a list of the currently installed environments by running

conda env list

Running conda list within an environment will list the packages you’ve installed within that environment.

When you install Miniconda, you will begin with a “base” environment. It’s a good idea not to use this for projects but to instead to create a new environment for each project.

There are two downsides to installing environments directly from the command line. One is that you may have lots of packages. The second is that you may wish to keep a record of the environment you created. For both of these reasons, you can specify a conda environment using a file.

The pandas example we saw above would look like

name: myenv
channels:
  - conda-forge
dependencies:
  - python=3.8
  - pandas
  - jupyter

The environment is given by name, the channel (where to look for the packages) by channels, the specific packages by dependencies. Not all packages are available on conda’s channels, so sometimes extra ones are needed. And some packages are only available on pip; these can be specified with a sub-section of the file like so for the skimpy package:

name: myenv
channels:
  - conda-forge
dependencies:
  - python=3.8
  - pandas
  - jupyter
  - pip:
    - skimpy

This goes into a file called environment.yml, which can be installed by running

conda env create -f environment.yml

This book is put together using an isolated conda environment specified in a file. It’s an unusually big one because there are a lot of packages, some of them with complex dependencies, featured in the book! Here they are:

Hide code cell source
from rich import print

with open("environment.yml", 'r') as stream:
    data_loaded = stream.read()

print(data_loaded)
name: codeforecon
channels:
  - conda-forge
dependencies:
  - jupyter
  - numpy<2.0.0
  - pandas>=2.1.0
  - pip
  - python=3.10
  - matplotlib
  - pyyaml
  - scipy
  - seaborn
  - statsmodels
  - yaml
  - pyarrow
  - geopandas
  - geoplot
  - nltk
  - arrow
  - osmnx
  - pandas-datareader
  - pandasdmx
  - pdfminer.six
  - jupyter-book
  - pytest
  - rich
  - pyinstrument
  - loguru
  - pre-commit
  - pandera
  - sympy
  - pingouin
  - plotly
  - spacy
  - plotnine
  - altair
  - cerberus
  - waterfallcharts
  - matplotlib-venn
  - pywaffle
  - joypy
  - jupyterlab
  - textstat
  - nbstripout
  - vega_datasets
  - ghp-import
  - colorcet
  - requests-cache
  - pip
  - arch-py
  - workalendar
  - wordcloud
  - bambi
  - bottleneck
  - openpyxl
  - python-slugify
  - pymc
  - umap-learn
  - python-dotenv
  - ruff
  - ruptures
  - great_tables
  - polars
  - ibis-sqlite
  - sqlmodel
  - scikit-learn==1.5.2
  - pip:
    - specification_curve
    - stargazer
    - matplotlib-scalebar
    - skimpy
    - graphviz
    - binsreg
    - feature-engine
    - lets-plot>=4.3.0
    - palmerpenguins
    - pyfixest>=0.17.0
    - watermark
    - ydata_profiling
    - toml-cli
    - toml

Of course, you can install packages as you go too, you don’t have to specify them when you create the environment. With the relevant environment activated, use conda install packagename to do this. However, for any project not involving complex packages like pymc, we recommend using uv instead.

Finally, to remove an environment, it’s

conda remove --name myenv --all

Using and Switching Between Conda Environments#

To switch between conda environments on the command line, for example from the base environment to an environment called “myenv”, use

conda activate myenv

on the command line. However, this only switches the environment if you plan to run code on the command line!

Fortunately, Visual Studio Code has you covered and makes it very easy to switch Python environments for a project at the click of a button.

A typical user view in Visual Studio Code

In the screenshot above, you can see the project-environment in two places: on the blue bar at the bottom of the screen, and (in 5), at the top right hand side of the interactive window. Click on either to change the Python environment that will be used to execute code. A similar top right selector is present for Jupyter Notebooks too.