Run, Host and Build Dependencies#
If you add a package to the dependency table of a feature that dependency will be available in all environments that include that feature. The dependencies of a package that is being built are a bit more granular. Here you can see the three types of dependencies for a simple C++ package.
[package.build-dependencies]
cxx-compiler = "*"
[package.host-dependencies]
catch = "*"
[package.run-dependencies]
git = "*"
Each dependency is used at a different step of the package building process.
cxx-compiler
is used to build the package, catch
will be linked into the package and git
will be available during runtime.
Let's delve deeper into the various types of package dependencies and their specific roles in the build process.
Build Dependencies#
pixi-build-cmake
When using the pixi-build-cmake
backend you do not need to specify cmake
or the compiler as a dependency.
The backend will install cmake
, ninja
and the C++ compilers by default.
This table contains dependencies that are needed to build the project. Different from dependencies and host-dependencies these packages are installed for the architecture of the build machine. This enables cross-compiling from one machine architecture to another.
Typical examples of build dependencies are:
- Compilers are invoked on the build machine, but they generate code for the target machine. If the project is cross-compiled, the architecture of the build and target machine might differ.
cmake
is invoked on the build machine to generate additional code- or project-files which are then include in the compilation process.
Info
The build target refers to the machine that will execute the build. Programs and libraries installed by these dependencies will be executed on the build machine.
For example, if you compile on a MacBook with an Apple Silicon chip but target Linux x86_64 then your build platform is osx-arm64
and your host platform is linux-64
.
Host Dependencies#
Host dependencies are the dependencies needed during build/link time that are specific to the host machine. The difference to build dependencies becomes for example important during cross compilation. The compiler is a build dependency since it is specific to your machine. In contrast, the libraries you link to are host dependencies since they are specific to the host machine. Typical examples of host dependencies are:
- Base interpreters: a Python package would list
python
here and an R package would listmro-base
orr-base
. - Libraries your project links against like
openssl
,rapidjson
, orxtensor
.
Python code#
Because of the way building currently works, dependencies like hatchling
,pip
,uv
etc. are host dependencies.
Otherwise, it would use the wrong python prefix during the build process.
This is more of a technical limitation, and we are looking into ways to make this less of a hassle.
But for now, you will need to add these dependencies to the host-dependencies
section.
So as an example, say we want to use hatchling
and uv
as to build a python package.
You need to use, something like this in your manifest file:
Native code#
When cross-compiling, you might need to specify host dependencies that should have the target machine architecture, and are used during the build process. When linking a library, for example. Let's recap an explanation that can be found here A Master Guide To Linux Cross-Compiling
- Build machine: where the code is built.
- Host machine: where the built code runs.
- Target machine: where the binaries spit out by the built code runs.
Lets say we are using a Linux PC (linux-64) to cross compile a CMake application called Awesome
to run on a Linux ARM target machine (linux-aarch64).
We would get the following table:
Component | Type | Build | Host | Target |
---|---|---|---|---|
GCC | Compiler | x86_64 | x86_64 | aarch64 |
CMake | Build tool | x86_64 | x86_64 | N/A |
Awesome | Application | x86_64 | aarch64 | N/A |
So if I need to use a library like SDL2, I would need to add it to the host-dependencies
table.
As the machine running Awesome
will have a different host architecture than the build architecture.
Giving you something like this in your manifest file:
Run-exports#
Conda packages, can define run-exports
, that are dependencies that when specified in the host-dependencies
section, will be implicitly be added to the run-dependencies
section.
This is useful to avoid having to specify the same dependencies in both sections.
As most packages on conda-forge will have these run-exports
defined.
When using something like zlib
, you would only need to specify it in the host-dependencies
section, and it will be used as a run-dependency automatically.
Dependencies (Run Dependencies)#
These are the dependencies that are required to when running the package, they are the most common dependencies.
And are what you would usually use in a workspace
.