Skip to content

hdl/conda-build-prepare

Repository files navigation

conda-build-prepare

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).

Installation

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.

Usage

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.

Building the example 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

Specifying additional channels

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.

Specifying additional packages

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.

Restoring original conda configurations

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

How conda-build-prepare works

The preparation process consists of:

  1. preparing the working directory,
  2. extracting the build environment information,
  3. preparing the build environment,
  4. rendering the recipe,
  5. cloning git source repositories,
  6. preparing git tags for better version description,
  7. embedding script_env variables.

Preparing the working directory

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.

Extracting the build environment information

Additional metadata is added based on the build environment:

  1. repository containing the recipe: its address, branch, commit and the result of git describe
  2. type of the environment: local build (default), Travis CI or Github Actions,
  3. additional information from Travis or GitHub Actions, such as event that started the build, job/run id etc.,
  4. TOOLCHAIN_ARCH it's being prepared for (if such variable is set),
  5. package's additional condarc contents (if any condarc is used).

Preparing the build environment

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.

Rendering the recipe

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.

Cloning git source repositories

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.

Preparing git tags for better version description

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.

Embedding script_env variables

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.

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages