Environments#
Pixi is a tool to manage virtual environments. This document explains what an environment looks like and how to use it.
Structure#
A pixi environment is located in the .pixi/envs
directory of the project by default.
This keeps your machine and your project clean and isolated from each other, and makes it easy to clean up after a project is done.
While this structure is generally recommended, environments can also be stored outside of project directories by enabling detached environments.
If you look at the .pixi/envs
directory, you will see a directory for each environment, the default
being the one that is normally used, if you specify a custom environment the name you specified will be used.
.pixi
└── envs
├── cuda
│ ├── bin
│ ├── conda-meta
│ ├── etc
│ ├── include
│ ├── lib
│ ...
└── default
├── bin
├── conda-meta
├── etc
├── include
├── lib
...
These directories are conda environments, and you can use them as such, but you cannot manually edit them, this should always go through the pixi.toml
.
Pixi will always make sure the environment is in sync with the pixi.lock
file.
If this is not the case then all the commands that use the environment will automatically update the environment, e.g. pixi run
, pixi shell
.
Environment Installation Metadata#
On environment installation, pixi will write a small file to the environment that contains some metadata about installation.
This file is called pixi
and is located in the conda-meta
folder of the environment.
This file contains the following information:
manifest_path
: The path to the manifest file that describes the project used to create this environmentenvironment_name
: The name of the environmentpixi_version
: The version of pixi that was used to create this environmentenvironment_lock_file_hash
: The hash of thepixi.lock
file that was used to create this environment
{
"manifest_path": "/home/user/dev/pixi/pixi.toml",
"environment_name": "default",
"pixi_version": "0.34.0",
"environment_lock_file_hash": "4f36ee620f10329d"
}
The environment_lock_file_hash
is used to check if the environment is in sync with the pixi.lock
file.
If the hash of the pixi.lock
file is different from the hash in the pixi
file, pixi will update the environment.
This is used to speedup activation, in order to trigger a full revalidation pass --revalidate
to the pixi run
or pixi shell
command.
A broken environment would typically not be found with a hash comparison, but a revalidation would reinstall the environment.
By default, all lock file modifying commands will always use the revalidation and on pixi install
it always revalidates.
Cleaning up#
If you want to clean up the environments, you can simply delete the .pixi/envs
directory, and pixi will recreate the environments when needed.
Activation#
An environment is nothing more than a set of files that are installed into a certain location, that somewhat mimics a global system install.
You need to activate the environment to use it.
In the most simple sense that mean adding the bin
directory of the environment to the PATH
variable.
But there is more to it in a conda environment, as it also sets some environment variables.
To do the activation we have multiple options:
- Use the
pixi shell
command to open a shell with the environment activated. - Use the
pixi shell-hook
command to print the command to activate the environment in your current shell. - Use the
pixi run
command to run a command in the environment.
Where the run
command is special as it runs its own cross-platform shell and has the ability to run tasks.
More information about tasks can be found in the tasks documentation.
Using the pixi shell-hook
in pixi you would get the following output:
export PATH="/home/user/development/pixi/.pixi/envs/default/bin:/home/user/.local/bin:/home/user/bin:/usr/local/bin:/usr/local/sbin:/usr/bin:/home/user/.pixi/bin"
export CONDA_PREFIX="/home/user/development/pixi/.pixi/envs/default"
export PIXI_PROJECT_NAME="pixi"
export PIXI_PROJECT_ROOT="/home/user/development/pixi"
export PIXI_PROJECT_VERSION="0.12.0"
export PIXI_PROJECT_MANIFEST="/home/user/development/pixi/pixi.toml"
export CONDA_DEFAULT_ENV="pixi"
export PIXI_ENVIRONMENT_PLATFORMS="osx-64,linux-64,win-64,osx-arm64"
export PIXI_ENVIRONMENT_NAME="default"
export PIXI_PROMPT="(pixi) "
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/activate-binutils_linux-64.sh"
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/activate-gcc_linux-64.sh"
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/activate-gfortran_linux-64.sh"
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/activate-gxx_linux-64.sh"
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/libglib_activate.sh"
. "/home/user/development/pixi/.pixi/envs/default/etc/conda/activate.d/rust.sh"
It sets the PATH
and some more environment variables. But more importantly it also runs activation scripts that are presented by the installed packages.
An example of this would be the libglib_activate.sh
script.
Thus, just adding the bin
directory to the PATH
is not enough.
You can modify the activation with the activation
table in the manifest, you can add more activation scripts or inject environment variables into the activation scripts.
[activation.env]
# Python users often set:
PYTHONIOENCODING = "utf-8"
PYTHONNOUSERSITE = "1"
# R users often set:
PIXI_R_LIBS = "$CONDA_PREFIX/lib/R/library"
R_LIBS = "$PIXI_R_LIBS"
R_LIBS_USER = "$PIXI_R_LIBS"
[target.unix.activation]
# Use sh scripts on unix
scripts = [
# Common in the ROS workspaces
"install/setup.sh",
# Want to add some personal scripts to the activation:
"activation.sh",
]
# Use batch scripts on windows
[target.win.activation]
scripts = ["install/setup.bat"]
activation
table here.
Traditional conda activate
-like activation#
If you prefer to use the traditional conda activate
-like activation, you could use the pixi shell-hook
command.
$ which python
python not found
$ eval "$(pixi shell-hook)"
$ (default) which python
/path/to/project/.pixi/envs/default/bin/python
Warning
It is not encouraged to use the traditional conda activate
-like activation, as deactivating the environment is not really possible. Use pixi shell
instead.
Using pixi
with direnv
#
Installing direnv
Of course you can use pixi
to install direnv
globally. We recommend to run
to install the latest version of direnv
on your computer.
This allows you to use pixi
in combination with direnv
.
Enter the following into your .envrc
file:
- This ensures that every time your
pixi.lock
changes,direnv
invokes the shell-hook again. - This installs if needed, and activates the environment.
direnv
ensures that the environment is deactivated when you leave the directory.
$ cd my-project
direnv: error /my-project/.envrc is blocked. Run `direnv allow` to approve its content
$ direnv allow
direnv: loading /my-project/.envrc
✔ Project in /my-project is ready to use!
direnv: export +CONDA_DEFAULT_ENV +CONDA_PREFIX +PIXI_ENVIRONMENT_NAME +PIXI_ENVIRONMENT_PLATFORMS +PIXI_PROJECT_MANIFEST +PIXI_PROJECT_NAME +PIXI_PROJECT_ROOT +PIXI_PROJECT_VERSION +PIXI_PROMPT ~PATH
$ which python
/my-project/.pixi/envs/default/bin/python
$ cd ..
direnv: unloading
$ which python
python not found
Environment variables#
The following environment variables are set by pixi, when using the pixi run
, pixi shell
, or pixi shell-hook
command:
PIXI_PROJECT_ROOT
: The root directory of the project.PIXI_PROJECT_NAME
: The name of the project.PIXI_PROJECT_MANIFEST
: The path to the manifest file (pixi.toml
).PIXI_PROJECT_VERSION
: The version of the project.PIXI_PROMPT
: The prompt to use in the shell, also used bypixi shell
itself.PIXI_ENVIRONMENT_NAME
: The name of the environment, defaults todefault
.PIXI_ENVIRONMENT_PLATFORMS
: Comma separated list of platforms supported by the project.CONDA_PREFIX
: The path to the environment. (Used by multiple tools that already understand conda environments)CONDA_DEFAULT_ENV
: The name of the environment. (Used by multiple tools that already understand conda environments)PATH
: We prepend thebin
directory of the environment to thePATH
variable, so you can use the tools installed in the environment directly.INIT_CWD
: ONLY INpixi run
: The directory where the command was run from.
Note
Even though the variables are environment variables these cannot be overridden. E.g. you can not change the root of the project by setting PIXI_PROJECT_ROOT
in the environment.
Solving environments#
When you run a command that uses the environment, pixi will check if the environment is in sync with the pixi.lock
file.
If it is not, pixi will solve the environment and update it.
This means that pixi will retrieve the best set of packages for the dependency requirements that you specified in the pixi.toml
and will put the output of the solve step into the pixi.lock
file.
Solving is a mathematical problem and can take some time, but we take pride in the way we solve environments, and we are confident that we can solve your environment in a reasonable time.
If you want to learn more about the solving process, you can read these:
Pixi solves both the conda
and PyPI
dependencies, where the PyPI
dependencies use the conda packages as a base, so you can be sure that the packages are compatible with each other.
These solvers are split between the rattler
and uv
library, these control the heavy lifting of the solving process, which is executed by our custom SAT solver: resolvo
.
resolve
is able to solve multiple ecosystem like conda
and PyPI
. It implements the lazy solving process for PyPI
packages, which means that it only downloads the metadata of the packages that are needed to solve the environment.
It also supports the conda
way of solving, which means that it downloads the metadata of all the packages at once and then solves in one go.
For the [pypi-dependencies]
, uv
implements sdist
building to retrieve the metadata of the packages, and wheel
building to install the packages.
For this building step, pixi
requires to first install python
in the (conda)[dependencies]
section of the pixi.toml
file.
This will always be slower than the pure conda solves. So for the best pixi experience you should stay within the [dependencies]
section of the pixi.toml
file.
Caching packages#
Pixi caches all previously downloaded packages in a cache folder. This cache folder is shared between all pixi projects and globally installed tools.
Normally the location would be the following platform-specific default cache folder:
- Linux:
$XDG_CACHE_HOME/rattler
or$HOME/.cache/rattler
- macOS:
$HOME/Library/Caches/rattler
- Windows:
%LOCALAPPDATA%\rattler
This location is configurable by setting the PIXI_CACHE_DIR
or RATTLER_CACHE_DIR
environment variable.
When you want to clean the cache, you can simply delete the cache directory, and pixi will re-create the cache when needed.
The cache contains multiple folders concerning different caches from within pixi.
pkgs
: Contains the downloaded/unpackedconda
packages.repodata
: Contains theconda
repodata cache.uv-cache
: Contains theuv
cache. This includes multiple caches, e.g.built-wheels
wheels
archives
http-cache
: Contains theconda-pypi
mapping cache.