This repository contains a tool for unified conda package building.
The conda-build-prepare
tool renders an dynamic package recipe into a resloved package recipe, conda environment and git repositories.
It is able to rendered metadata and embedding crucial data used inside a dynamic package recipe. Doing this rendering makes the package building much more consistent regardless of when or in which environment the building is done.
The data is also extracted from the repository containing the recipe to populate the recipe_append.yaml
file, which is automatically added to the metadata (as of conda-build
v3.0).
Using pip
is the easiest way to install conda-build-prepare
:
python3 -m pip install git+https://github.com/hdl/conda-build-prepare@v0.1.2#egg=conda-build-prepare
where v0.1.2
can be replaced by any tag or revision known to git.
Alternatively, python3 setup.py install
can be called after cloning the repository.
Either way other packages required by conda-build-prepare
will be installed automatically.
After installing conda-build-prepare
run:
python3 -m conda_build_prepare --dir $DIRECTORY -- $RECIPE
to prepare the RECIPE and store output files in a DIRECTORY without specifying any additional channels or packages.
The RECIPE argument needs to be separated with --
in case it is preceded by some optional --channels
or --packages
argument.
After running the tool, the package can be built using the prepared environment and recipe using the following commands:
conda activate $DIRECTORY/conda-env
conda build $DIRECTORY/recipe
conda deactivate # restores previous conda environment
DIRECTORY can be virtually any name but it has to match the --dir
argument passed to prepare the package.
wishbone-tool
recipe is included for testing in this repository in the test
directory.
The package can be successfully built with the aforementioned commands after specifying the recipe path with the RECIPE
variable and using any DIRECTORY
, e.g.:
RECIPE=test/wishbone-tool
DIRECTORY=X
Additional channels can be specified with the --channels CHANNEL [CHANNEL ...]
option.
Each CHANNEL
will be added to the $DIRECTORY/conda-env/.condarc
.
The last one used will be the most important channel for conda-build
during preparation and building.
Additional packages can be specified with the --packages PACKAGE [PACKAGE ...]
option.
Each PACKAGE
will be installed in the created $DIRECTORY/conda-env
environment.
The conda-build
package is installed even if not specified with this option but it can be used to specify some precise version, e.g. --packages conda-build=3.20.3
.
conda-build-prepare
"neutralizes" conda configuration files by commenting them out to make environment as reproducible as it's possible.
The original configuration can be restored later using:
python3 -m conda_build_prepare restore
The preparation process consists of:
- preparing the working directory,
- extracting the build environment information,
- preparing the build environment,
- rendering the recipe,
- cloning git source repositories,
- preparing git tags for better version description,
- embedding script_env variables.
The argument passed to --dir
($DIRECTORY
) is used as a target directory name, in which the conda-env
, git-repos
and recipe
directories will be created.
The directory specified as a $RECIPE
will be copied as the $DIRECTORY/recipe
directory.
While all packages are expected to have a meta.yaml
, a prescript file (prescript.${TOOLCHAIN_ARCH}.sh
) can be used to download or generate it.
Prescript file is executed right after copying the $RECIPE
.
Additional metadata is added based on the build environment:
- repository containing the recipe: its address, branch, commit and the result of
git describe
- type of the environment: local build (default), Travis CI or Github Actions,
- additional information from Travis or GitHub Actions, such as event that started the build, job/run id etc.,
TOOLCHAIN_ARCH
it's being prepared for (if such variable is set),- package's additional
condarc
contents (if anycondarc
is used).
Conda environment created in $DIRECTORY/conda-env
will contain basic packages necessary to build and render metadata.
Specifically, those basic packages are: anaconda-client
, conda-build
, conda-verify
, jinja2
, pexpect
, python
and ripgrep
(ripgrep
only on Linux and macOS).
In the next step, conda-build-prepare
will look for all condarc
files affecting the newlyy created environment.
All such files found by the tool will be "neutralized" by commenting them out.
Paths are added to the conda-build-prepare_srcs.txt
file inside the system's temp dir (tempfile.gettempdir()
) for a possbible future restoration of the files, which can be triggered by the user.
Then, the package's condarc (condarc
, condarc_linux
, condarc_macos
or condarc_windows
from $RECIPE
) will be set as the most important one (conda-env/condarc
) and other basic settings will be applied to that environment.
The goal is to set each package used for building with specific version.
This will allow using the same packages for building even if any of the required build
or host
package gets updated in the channels.
Conda environment created in $DIRECTORY/conda-env
is used for rendering the recipe to ensure the same settings for future building (channels, channel priority etc.).
The rendered version of the recipe will replace the original meta.yaml
copied from the $RECIPE
.
Contents of the original meta.yaml
will be left at the end of the new file as a comment.
recipe_append.yaml
is incorporated in the rendered recipe automatically by the conda render
.
Due to problems with Conda's own repository management [TODO: what exactly?], conda-build-prepare
clones the repository and changes the recipe for conda-build
to operate on the cloned one.
Each git_url
source repository is cloned to the $DIRECTORY/git-repos
directory.
The relative submodules of those repositories (where submodule's url starts with ../
) will also get cloned because git
will search for them in the same parent directory during the building procedure.
The resulting recipe will have each git_url
replaced with the local path to a repository cloned from the original repository URL.
The previous version of conda-related tooling used to rewrite git tags in order to let conda automatically detect the version via git-describe
.
The conda-build-prepare
tool makes version format unified across various packages.
This is achieved by checking recipe source's git tags for any version-like part and modifying it by leaving only this version-like part prefixed with a v
after rewriting.
The following python code describes supported version formats best:
version_spec = r"""[0-9]+[_.\-][0-9]+ # required major and minor ([_.\-][0-9]+)? # optional micro ([_.\-][0-9]+)? # optional extra number ([._\-]*rc[0-9]+)? # optional release candidate"""
Therefore, the version specifier consists of two to four numbers separated from each other and an optional release candidate number after rc
which can be separated (e.g. 2.1-rc2
).
Each separator can be an underscore, a point or a dash.
If no valid tags are found, a v0.0
tag is created on the oldest commit in the repository.
Finally, the package version will be set with the git describe
result on such a repository.
Any dashes in version will be replaced by underscores because of the conda's restrictions in setting the package/version
key.
The recipe can allow some environment variables to influence building through the build/script_env
key.
To make building process reproducible, conda-build-prepare
embeds all such variables inside the conda-env
with the values found in the current shell environment during preparation.
Such embedded variables will be later set in the shell while activating conda environment.
If during this stage there are some script_env
variables not set to any value in shell, they will be removed from the script_env
to never affect building of this package.