Tutorial: Doing python development with pixi#
In this tutorial we will show you how to create a simple python project with pixi. We will show some of the features that pixi provides, that are currently
not a part of pdm
, poetry
etc.
Why is this useful?#
Pixi builts upon on the conda ecosystem, which allows you to create a python environment with all the dependencies you need. This is especially useful when you are working with multiple python interpreters and bindings to C and C++ libraries. Let's give it a go! E.g GDAL from PyPI does not have the binary C dependencies, but the conda package does.
pixi.toml and pyproject.toml#
We support two manifests formats, pyproject.toml
and pixi.toml
. In this tutorial we will use the pyproject.toml
format. Because it is the most common format for python projects.
Let's get started#
Lets start out by making a directory and creating a new pyproject.toml
file.
This gives you the following pyproject.toml:
[project]
name = "pixi_py"
version = "0.1.0"
description = "Add a short description here"
authors = [{name = "Tim de Jager", email = "tim@prefix.dev"}]
requires-python = ">= 3.11"
dependencies = []
[build-system]
requires = ["setuptools"]
build-backend = "setuptools.build_meta"
[tool.pixi.project]
channels = ["conda-forge"]
platforms = ["osx-arm64"]
[tool.pixi.pypi-dependencies]
pixi_py = { path = ".", editable = true }
[tool.pixi.tasks]
Let's add the python project to the tree:
We now have the following directory structure:
We've used a flat-layout here but we support both flat- and src-layouts.
What's in the pyproject?#
Okay, so let's have a look at what's sections have been added.
These things were added to the pyproject.toml file:
# 1. Main pixi entry
[tool.pixi.project]
channels = ["conda-forge"]
# This is your machine platform by default
platforms = ["osx-arm64"]
# 2. Editable installs
[tool.pixi.pypi-dependencies]
pixi_py = { path = ".", editable = true }
Let's go over them one by one:
- The
channels
andplatforms
are added to the[tool.pixi.project]
section. Channels likeconda-forge
manage packages similar to pypi but allow for different packages across languages. Platforms determines for what platform to generate the lock for. - The
pixi_py
package is added as an editable dependency. This means that the package is installed in editable mode, so you can make changes to the package and see the changes reflected in the environment. In pixi, unlike other package managers this is explicitly stated in thepyproject.toml
file. The main reason being so that you can choose into which environment this package should be included.
Now let's add some optional dependencies:
[project.optional-dependencies]
test = ["pytest"]
# 3. Environments
[tool.pixi.environments]
default = { solve-group = "default" }
test = { features = ["test"], solve-group = "default" }
We've added the optional dependencies to the pyproject.toml
and this automatically creates a feature
, which is a collection of dependencies
, tasks
, channels
etc.
These features can be combined
- We've created the
default
andtest
environments. Thedefault
environment is the default environment that is created when you runpixi install
. Thetest
environment is created from the optional dependencies in thepyproject.toml
file. You can execute commands in this environment with e.gpixi run -e test python
Solve Groups
Solve groups are a way to group dependencies together. This is useful when you have multiple environments that share the same dependencies. This way you can solve dependencies together. E.g maybe pytest
is a dependency that influences the dependencies of the default
environment. By putting these in the same solve group, you assure that the versions in test
and default
are exactly the same versions.
Installation: pixi install
#
Now let's install
the project with pixi install
:
We now have a new directory called .pixi
in the project root. This directory contains the environment that was created when we ran pixi install
. The environment is a conda environment that contains the dependencies that we specified in the pyproject.toml
file. We can also install the test environment with pixi install -e test
. We can use these environments for executing code.
We also have a new file called pixi.lock
in the project root. This file contains the exact versions of the dependencies that were installed in the environment, across platforms.
What's in the environment?#
Python
The python interpreter is also installed in the environment. This is because the python interpreter version is read from the requires-python
field in the pyproject.toml
file. This is used to determine the python version to install in the environment. This way pixi automatically manages/bootstraps the python interpreter for you, so no more brew
, apt
or other system install steps.
Using pixi list
you can see what's in the environment, this is essentially a nicer view on the lock file:
$ pixi list
Package Version Build Size Kind Source
bzip2 1.0.8 h93a5062_5 119.5 KiB conda bzip2-1.0.8-h93a5062_5.conda
ca-certificates 2024.2.2 hf0a4a13_0 152.1 KiB conda ca-certificates-2024.2.2-hf0a4a13_0.conda
libexpat 2.6.2 hebf3989_0 62.2 KiB conda libexpat-2.6.2-hebf3989_0.conda
libffi 3.4.2 h3422bc3_5 38.1 KiB conda libffi-3.4.2-h3422bc3_5.tar.bz2
libsqlite 3.45.2 h091b4b1_0 806 KiB conda libsqlite-3.45.2-h091b4b1_0.conda
libzlib 1.2.13 h53f4e23_5 47 KiB conda libzlib-1.2.13-h53f4e23_5.conda
ncurses 6.4.20240210 h078ce10_0 801 KiB conda ncurses-6.4.20240210-h078ce10_0.conda
openssl 3.2.1 h0d3ecfb_1 2.7 MiB conda openssl-3.2.1-h0d3ecfb_1.conda
python 3.12.3 h4a7b5fc_0_cpython 12.6 MiB conda python-3.12.3-h4a7b5fc_0_cpython.conda
readline 8.2 h92ec313_1 244.5 KiB conda readline-8.2-h92ec313_1.conda
tk 8.6.13 h5083fa2_1 3 MiB conda tk-8.6.13-h5083fa2_1.conda
tzdata 2024a h0c530f3_0 117 KiB conda tzdata-2024a-h0c530f3_0.conda
pixi_py 0.1.0 pypi . (editable)
xz 5.2.6 h57fd34a_0 230.2 KiB conda xz-5.2.6-h57fd34a_0.tar.bz2
Here you can see the different conda and pypi packages listed. And as you can see the pixi_py
package, that we are working on, is installed in editable mode. Every environment in pixi is isolated, but re-uses files that are hard-linked from a central cache directory. This means that you can have multiple environments with the same packages, but only have the individual files stored once on disk.
Getting code to run#
Lets add some code to the pixi_py
package. We will add a new function to the pixi_py/__init__.py
file:
from rich import print
def hello():
return "Hello, [bold magenta]World[/bold magenta]!", ":vampire:"
def say_hello():
print(*hello())
rich
dependency using: pixi add --pypi rich
.
Let's see if it this works by running:
Slow?
The first time this might be slow because we install the project but the second time it will be instant.
So we are running a python interpreter installed from conda. And we are importing the pixi_py
package that is installed in editable mode. And we are calling the say_hello
function that we just added. And it works! Cool!
Testing this code#
Okay, so let add a test for this function. Let's add a tests/test_me.py
file in root of the project.
Giving us the following project structure:
from pixi_py import hello
def test_pixi_py():
assert hello() == ("Hello, [bold magenta]World[/bold magenta]!", ":vampire:")
Let's add an easy task for running the tests.
So pixi has a task system to make it easy to run commands. Similar to npm
scripts or something you would specify in a Justfile
.
Pixi tasks
Tasks are actually a pretty cool pixi feature that are powerful and are run in a cross-platform shell. You can do caching, dependencies and more. Read more about tasks in the tasks section.
$ pixi r test
✨ Pixi task (test): pytest .
================================================================================================= test session starts =================================================================================================
platform darwin -- Python 3.12.2, pytest-8.1.1, pluggy-1.4.0
rootdir: /private/tmp/pixi_py
configfile: pyproject.toml
collected 1 item
test_me.py . [100%]
================================================================================================== 1 passed in 0.00s =================================================================================================
Cool it seems to be working!
Test vs Default environment#
Interesting thing is if we compare the output of the two environments.
Is that the test environment has:
package version build size kind source
...
pytest 8.1.1 1.1 mib pypi pytest-8.1.1-py3-none-any.whl
...
But the default environment is missing this package.
This way you can finetune your environments to only have the packages that are needed for that environment.
E.g you could also have a dev
environment that has pytest
and ruff
installed, but you could omit these from the prod
environment.
There is a docker example that shows how to set-up a minimal prod
environment, and copy from there.
Replacing pypi packages with conda packages#
Last thing, pixi provides the ability for pypi
packages to depend on conda
packages.
Let's add pygments
to the pyproject.toml
file. Which is a dependency of the rich
package.
pyproject.toml
file:
We can now see that the pygments
package is now installed as conda package.
$ pixi list
Package Version Build Size Kind Source
...
pygments 2.17.2 pyhd8ed1ab_0 840.3 KiB conda pygments-2.17.2-pyhd8ed1ab_0.conda
This way PyPI dependencies and conda dependencies can be mixed and matched, and seamlessly interoperate.
And it still works!
Conclusion#
In this tutorial you've seen how easy it is to use a pyproject.toml
to manage your pixi dependencies. As well, as how to use PyPI and conda dependencies together in the same project.
Hopefully, this provides for flexible and powerful way to manage your python projects, and a fertile base for further exploration with pixi.
Thanks for reading! Happy Coding 🚀
Any questions? Feel free to reach out or share this tutorial on X, join our Discord, send us an e-mail or follow our GitHub.