diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 966d7cb2da5d2323b995b8e0edfe62edc916e022..941df9c4d901fc2de8106e3bf3532f854a3a61b5 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -23,7 +23,7 @@ stages: - ${PRE_COMMIT_HOME} -build_linux: +.build_linux_template: extends: .build_template variables: BUILD_EGG: "true" @@ -33,13 +33,9 @@ build_linux: before_script: - rm -f /root/.condarc - rm -rf /root/.conda - script: - python3 ./bob/devtools/bootstrap.py -vv build - source ${CONDA_ROOT}/etc/profile.d/conda.sh - conda activate base - - pip install pre-commit - - pre-commit run --all-files --show-diff-on-failure - - python3 ./bob/devtools/build.py -vv --twine-check artifacts: paths: - dist/*.zip @@ -52,18 +48,15 @@ build_linux: key: "linux-cache" -build_macos_intel: +.build_macos_intel_template: extends: .build_template tags: - macos - intel - script: + before_script: - python3 ./bob/devtools/bootstrap.py -vv build - source ${CONDA_ROOT}/etc/profile.d/conda.sh - conda activate base - - pip install pre-commit - - pre-commit run --all-files --show-diff-on-failure - - python3 ./bob/devtools/build.py -vv artifacts: paths: - ${CONDA_ROOT}/conda-bld/osx-64/*.conda @@ -72,6 +65,41 @@ build_macos_intel: key: "macos-intel-cache" +build_linux_bob_devel: + extends: .build_linux_template + script: + - python3 ./bob/devtools/build.py -vv build-bob-devel + +build_linux_deps: + extends: .build_linux_template + script: + - python3 ./bob/devtools/build.py -vv build-deps + +build_linux_bob_devtools: + extends: .build_linux_template + script: + - pip install pre-commit + - pre-commit run --all-files --show-diff-on-failure + - python3 ./bob/devtools/build.py -vv build-devtools --twine-check + +build_macos_intel_bob_devel: + extends: .build_macos_intel_template + script: + - python3 ./bob/devtools/build.py -vv build-bob-devel + +build_macos_intel_deps: + extends: .build_macos_intel_template + script: + - python3 ./bob/devtools/build.py -vv build-deps + +build_macos_intel_bob_devtools: + extends: .build_macos_intel_template + script: + - pip install pre-commit + - pre-commit run --all-files --show-diff-on-failure + - python3 ./bob/devtools/build.py -vv build-devtools + + # Deploy targets .deploy_template: stage: deploy diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 2357f8b3aa4e1686574bdab3856b054d5baba8e3..7bb4121fd20252cd89eb1672894bfc8da69844fa 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -15,7 +15,11 @@ repos: rev: 3.9.2 hooks: - id: flake8 - exclude: bob/devtools/templates/setup.py + exclude: | + (?x)^( + bob/devtools/templates/setup.py| + deps/bob-devel/run_test.py + )$ - repo: https://github.com/pre-commit/pre-commit-hooks rev: v4.0.1 hooks: @@ -29,4 +33,4 @@ repos: - id: check-added-large-files exclude: bob/devtools/templates/setup.py - id: check-yaml - exclude: .*/meta.yaml + exclude: .*/meta.*.yaml diff --git a/bob/devtools/bootstrap.py b/bob/devtools/bootstrap.py index e4180143aa8f34849687e52b408f418ba726f253..456d884ca109a6dd3912ccf487e7c7b52c2594b7 100644 --- a/bob/devtools/bootstrap.py +++ b/bob/devtools/bootstrap.py @@ -520,6 +520,7 @@ if __name__ == "__main__": "conda=%s" % conda_version, "conda-build=%s" % conda_build_version, "conda-verify=%s" % conda_verify_version, + "click", "twine", # required for checking readme of python (zip) distro ] ) diff --git a/bob/devtools/build.py b/bob/devtools/build.py index 51331dbc68395b27c3dabfbadc89816420499df0..1b62f3a2ebc01997df6dbe52bea5eb462d12a4ca 100644 --- a/bob/devtools/build.py +++ b/bob/devtools/build.py @@ -6,7 +6,6 @@ import contextlib import copy -import datetime import distutils.version import glob import json @@ -16,8 +15,8 @@ import platform import re import subprocess import sys -import tempfile +import click import conda_build.api import yaml @@ -705,8 +704,14 @@ def base_build( return conda_build.api.build(recipe_dir, config=conda_config) -def global_pin( - bootstrap, server, intranet, group, conda_build_config, condarc_options +def bob_devel( + bootstrap, + server, + intranet, + group, + conda_build_config, + condarc_options, + work_dir, ): """ Tests that all packages listed in bob/devtools/data/conda_build_config.yaml @@ -727,198 +732,84 @@ def global_pin( packages = [package_names_map.get(p, p) for p in package_pins.keys()] - with tempfile.TemporaryDirectory() as tmpdir: - # Create a conda-build recipe - recipe_path = tmpdir + "/meta.yaml" - with open(recipe_path, "w") as f: - content = """ -package: - name: global-pins - version: {date} - -build: - number: 0 - -requirements: - host: - - python {{{{ python }}}} - - {{{{ compiler('c') }}}} - - {{{{ compiler('cxx') }}}} -{package_list} - - run: - - python - {{% for package in resolved_packages('host') %}} - - {{{{ package }}}} - {{% endfor %}} - -test: - requires: - - numpy - - ffmpeg - - pytorch - - torchvision - - setuptools - commands: - # we expect these features from ffmpeg: - - ffmpeg -codecs | grep "DEVI.S zlib" # [unix] - - ffmpeg -codecs | grep "DEV.LS h264" # [unix] -""".format( - date=datetime.date.today().strftime("%Y.%m.%d"), - package_list="\n".join( - [ - " - {p1} {{{{ {p2} }}}}".format( - p1=p, p2=p.replace("-", "_").replace(".", "_") - ) - for p in packages - ] - ), - ) - logger.info( - "Writing a conda build recipe with the following content:\n%s", - content, - ) - f.write(content) - - # write run_test.py file - run_test_path = tmpdir + "/run_test.py" - with open(run_test_path, "w") as f: - content = """ -import sys - -# couple of imports to see if packages are working -import numpy -import pkg_resources - - -def test_pytorch(): - import torch - from torchvision.models import DenseNet - - model = DenseNet() - t = torch.randn(1, 3, 224, 224) - out = model(t) - assert out.shape[1] == 1000 - - -def _check_package(name, pyname=None): - "Checks if a Python package can be `require()`'d" - - pyname = pyname or name - print(f"Checking Python setuptools integrity for {name} (pyname: {pyname})") - pkg_resources.require(pyname) - - -def test_setuptools_integrity(): - - _check_package('pytorch', 'torch') - _check_package('torchvision') - - -# test if pytorch installation is sane -test_pytorch() -test_setuptools_integrity() -""" - logger.info( - "Writing a run_test.py file with the following content:\n%s", - content, + recipe_dir = os.path.join(work_dir, "deps", "bob-devel") + template_yaml = os.path.join(recipe_dir, "meta.template.yaml") + final_yaml = os.path.join(recipe_dir, "meta.yaml") + package_list = "\n".join( + [ + " - {p1} {{{{ {p2} }}}}".format( + p1=p, p2=p.replace("-", "_").replace(".", "_") ) - f.write(content) + for p in packages + ] + ) - # run conda build - base_build( - bootstrap=bootstrap, - server=server, - intranet=intranet, - group=group, - recipe_dir=tmpdir, - conda_build_config=conda_build_config, - condarc_options=condarc_options, + with open(template_yaml) as fr, open(final_yaml, "w") as fw: + content = fr.read() + content = content.replace("# PACKAGE_LIST", package_list) + logger.info( + "Writing a conda build recipe with the following content:\n%s", + content, ) + fw.write(content) - -if __name__ == "__main__": - - import argparse - - parser = argparse.ArgumentParser( - description="Builds bob.devtools on the CI" - ) - parser.add_argument( - "-g", - "--group", - default=os.environ.get("CI_PROJECT_NAMESPACE", "bob"), - help="The namespace of the project being built [default: %(default)s]", - ) - parser.add_argument( - "-n", - "--name", - default=os.environ.get("CI_PROJECT_NAME", "bob.devtools"), - help="The name of the project being built [default: %(default)s]", - ) - parser.add_argument( - "-c", - "--conda-root", - default=os.environ.get( - "CONDA_ROOT", os.path.realpath(os.path.join(os.curdir, "miniconda")) - ), - help="The location where we should install miniconda " - "[default: %(default)s]", - ) - parser.add_argument( - "-V", - "--visibility", - choices=["public", "internal", "private"], - default=os.environ.get("CI_PROJECT_VISIBILITY", "public"), - help="The visibility level for this project [default: %(default)s]", - ) - parser.add_argument( - "-t", - "--tag", - default=os.environ.get("CI_COMMIT_TAG", None), - help="If building a tag, pass it with this flag [default: %(default)s]", - ) - parser.add_argument( - "-w", - "--work-dir", - default=os.environ.get("CI_PROJECT_DIR", os.path.realpath(os.curdir)), - help="The directory where the repo was cloned [default: %(default)s]", - ) - parser.add_argument( - "-T", - "--twine-check", - action="store_true", - default=False, - help="If set, then performs the equivalent of a " - '"twine check" on the generated python package (zip file)', - ) - parser.add_argument( - "--internet", - "-i", - default=False, - action="store_true", - help="If executing on an internet-connected server, unset this flag", - ) - parser.add_argument( - "--verbose", - "-v", - action="count", - default=0, - help="Increases the verbosity level. We always prints error and " - "critical messages. Use a single ``-v`` to enable warnings, " - "two ``-vv`` to enable information messages and three ``-vvv`` " - "to enable debug messages [default: %(default)s]", - ) - parser.add_argument( - "--test-mark-expr", - "-A", - default="", - help="Use this flag to avoid running certain tests during the build. " - "It forwards all settings to ``nosetests`` via --eval-attr=<settings>``" - " and ``pytest`` via -m=<settings>.", + # run conda build + packages = base_build( + bootstrap=bootstrap, + server=server, + intranet=intranet, + group=group, + recipe_dir=recipe_dir, + conda_build_config=conda_build_config, + condarc_options=condarc_options, ) - args = parser.parse_args() + print(f"The following packages were built: {packages}") + + +@click.group() +@click.option( + "-g", + "--group", + default=os.environ.get("CI_PROJECT_NAMESPACE", "bob"), + help="The namespace of the project being built", +) +@click.option( + "-c", + "--conda-root", + default=os.environ.get( + "CONDA_ROOT", os.path.realpath(os.path.join(os.curdir, "miniconda")) + ), + help="The location where we should install miniconda", +) +@click.option( + "-w", + "--work-dir", + default=os.environ.get("CI_PROJECT_DIR", os.path.realpath(os.curdir)), + help="The directory where the repo was cloned", +) +@click.option( + "--internet", + "-i", + is_flag=True, + help="If executing on an internet-connected server, unset this flag", +) +@click.option( + "--verbose", + "-v", + default=0, + help="Increases the verbosity level. We always prints error and critical messages. Use a single '-v' to enable warnings, two '-vv' to enable information messages and three '-vvv' to enable debug messages [default: %(default)s]", +) +@click.option( + "--test-mark-expr", + "-A", + default="", + help="Use this flag to avoid running certain tests during the build. It forwards all settings to 'nosetests' via --eval-attr=<settings> and 'pytest' via -m=<settings>.", +) +@click.pass_context +def cli(ctx, group, conda_root, work_dir, internet, verbose, test_mark_expr): + "Builds bob.devtools on the CI" + ctx.ensure_object(dict) # loads the "adjacent" bootstrap module import importlib.util @@ -930,25 +821,20 @@ if __name__ == "__main__": spec.loader.exec_module(bootstrap) server = bootstrap._SERVER - bootstrap.setup_logger(logger, args.verbose) + bootstrap.setup_logger(logger, verbose) bootstrap.set_environment("DOCSERVER", server) bootstrap.set_environment("LANG", "en_US.UTF-8") bootstrap.set_environment("LC_ALL", os.environ["LANG"]) - bootstrap.set_environment("NOSE_EVAL_ATTR", args.test_mark_expr) - bootstrap.set_environment("PYTEST_ADDOPTS", f"-m '{args.test_mark_expr}'") - - # get information about the version of the package being built - version, is_prerelease = check_version(args.work_dir, args.tag) - bootstrap.set_environment("BOB_PACKAGE_VERSION", version) + bootstrap.set_environment("NOSE_EVAL_ATTR", test_mark_expr) + bootstrap.set_environment("PYTEST_ADDOPTS", f"-m '{test_mark_expr}'") # create the build configuration conda_build_config = os.path.join( - args.work_dir, "conda", "conda_build_config.yaml" + work_dir, "conda", "conda_build_config.yaml" ) - recipe_append = os.path.join(args.work_dir, "data", "recipe_append.yaml") - condarc = os.path.join(args.conda_root, "condarc") + condarc = os.path.join(conda_root, "condarc") logger.info("Loading (this build's) CONDARC file from %s...", condarc) with open(condarc, "rb") as f: condarc_options = yaml.load(f, Loader=yaml.FullLoader) @@ -958,21 +844,48 @@ if __name__ == "__main__": if condarc_options.get("conda-build", {}).get("root-dir") is None: condarc_options["croot"] = os.path.join(prefix, "conda-bld") - # Check global pin versions compatibility - global_pin( - bootstrap=bootstrap, - server=server, - intranet=not args.internet, - group=args.group, + # populate ctx.obj + ctx.obj["test_mark_expr"] = test_mark_expr + + ctx.obj["verbose"] = verbose + ctx.obj["conda_root"] = conda_root + ctx.obj["group"] = group + ctx.obj["bootstrap"] = bootstrap + ctx.obj["server"] = server + ctx.obj["work_dir"] = work_dir + ctx.obj["internet"] = internet + ctx.obj["condarc_options"] = condarc_options + ctx.obj["conda_build_config"] = conda_build_config + + +@cli.command() +@click.pass_obj +def build_bob_devel(obj): + bob_devel( + bootstrap=obj["bootstrap"], + server=obj["server"], + intranet=not obj["internet"], + group=obj["group"], conda_build_config=os.path.join( - args.work_dir, "bob", "devtools", "data", "conda_build_config.yaml" + obj["work_dir"], + "bob", + "devtools", + "data", + "conda_build_config.yaml", ), - condarc_options=condarc_options, + condarc_options=obj["condarc_options"], ) - # builds all dependencies in the 'deps' subdirectory - or at least checks - # these dependencies are already available; these dependencies go directly - # to the public channel once built + git_clean_build(obj["bootstrap"].run_cmdline, verbose=(obj["verbose"] >= 3)) + + +@cli.command() +@click.pass_obj +def build_deps(obj): + """builds all dependencies in the 'deps' subdirectory - or at least checks + these dependencies are already available; these dependencies go directly + to the public channel once built + """ recipes = load_order_file(os.path.join("deps", "order.txt")) for k, recipe in enumerate([os.path.join("deps", k) for k in recipes]): @@ -980,22 +893,61 @@ if __name__ == "__main__": # ignore - not a conda package continue base_build( - bootstrap, - server, - not args.internet, - args.group, + obj["bootstrap"], + obj["server"], + not obj["internet"], + obj["group"], recipe, - conda_build_config, - condarc_options, + obj["conda_build_config"], + obj["condarc_options"], ) - public = args.visibility == "public" + git_clean_build(obj["bootstrap"].run_cmdline, verbose=(obj["verbose"] >= 3)) + + +@cli.command() +@click.option( + "-T", + "--twine-check", + is_flag=True, + help="If set, then performs the equivalent of a 'twine check' on the generated python package (zip file)", +) +@click.option( + "-V", + "--visibility", + type=click.Choice(["public", "internal", "private"]), + default=os.environ.get("CI_PROJECT_VISIBILITY", "public"), + help="The visibility level for this project", +) +@click.option( + "-t", + "--tag", + default=os.environ.get("CI_COMMIT_TAG"), + help="If building a tag, pass it with this flag", +) +@click.pass_obj +def build_devtools(obj, twine_check, visibility, tag): + bootstrap = obj["bootstrap"] + condarc_options = obj["condarc_options"] + conda_build_config = obj["conda_build_config"] + work_dir = obj["work_dir"] + server = obj["server"] + internet = obj["internet"] + group = obj["group"] + + # get information about the version of the package being built + version, is_prerelease = check_version(work_dir, tag) + bootstrap.set_environment("BOB_PACKAGE_VERSION", version) + + recipe_append = os.path.join(work_dir, "data", "recipe_append.yaml") + + public = visibility == "public" channels, upload_channel = bootstrap.get_channels( public=public, stable=(not is_prerelease), server=server, - intranet=(not args.internet), - group=args.group, + intranet=(not internet), + group=group, ) if "channels" not in condarc_options: @@ -1010,19 +962,19 @@ if __name__ == "__main__": conda_build_config, None, recipe_append, condarc_options ) - recipe_dir = os.path.join(args.work_dir, "conda") + recipe_dir = os.path.join(obj["work_dir"], "conda") metadata = get_rendered_metadata(recipe_dir, conda_config) paths = get_output_path(metadata, conda_config) # asserts we're building at the right location for path in paths: - assert path.startswith(os.path.join(args.conda_root, "conda-bld")), ( + assert path.startswith(os.path.join(obj["conda_root"], "conda-bld")), ( 'Output path for build (%s) does not start with "%s" - this ' "typically means this build is running on a shared builder and " "the file ~/.conda/environments.txt is polluted with other " "environment paths. To fix, empty that file and set its mode " "to read-only for all." - % (path, os.path.join(args.conda_root, "conda-bld")) + % (path, os.path.join(obj["conda_root"], "conda-bld")) ) # retrieve the current build number(s) for this build @@ -1034,8 +986,6 @@ if __name__ == "__main__": build_number = max([int(k) for k in build_numbers]) # runs the build using the conda-build API - arch = conda_arch() - # notice we cannot build from the pre-parsed metadata because it has already # resolved the "wrong" build number. We'll have to reparse after setting the # environment variable BOB_BUILD_NUMBER. @@ -1044,7 +994,7 @@ if __name__ == "__main__": conda_build.api.build(recipe_dir, config=conda_config) # checks if long_description of python package renders fine - if args.twine_check: + if twine_check: from twine.commands.check import check package = glob.glob("dist/*.zip") @@ -1057,4 +1007,8 @@ if __name__ == "__main__": else: logger.info("twine check (a.k.a. readme check) %s: OK", package[0]) - git_clean_build(bootstrap.run_cmdline, verbose=(args.verbose >= 3)) + git_clean_build(bootstrap.run_cmdline, verbose=(obj["verbose"] >= 3)) + + +if __name__ == "__main__": + cli(obj={}) diff --git a/deps/bob-devel/meta.template.yaml b/deps/bob-devel/meta.template.yaml new file mode 100644 index 0000000000000000000000000000000000000000..8f0e73b79bda52809371da334735cb58223be5e9 --- /dev/null +++ b/deps/bob-devel/meta.template.yaml @@ -0,0 +1,32 @@ +package: + name: bob-devel + version: 2021.09.14 + +build: + number: 0 + +requirements: + host: + - python {{ python }} + - {{ compiler('c') }} + - {{ compiler('cxx') }} +# PACKAGE_LIST + + run: + - python + run_constrained: + {% for package in resolved_packages('host') %} + - {{ package }} + {% endfor %} + +test: + requires: + - numpy + - ffmpeg + - pytorch + - torchvision + - setuptools + commands: + # we expect these features from ffmpeg: + - ffmpeg -codecs | grep "DEVI.S zlib" # [unix] + - ffmpeg -codecs | grep "DEV.LS h264" # [unix] diff --git a/deps/bob-devel/run_test.py b/deps/bob-devel/run_test.py new file mode 100644 index 0000000000000000000000000000000000000000..1f0098f7488339391b68417d35744d83725aa5dc --- /dev/null +++ b/deps/bob-devel/run_test.py @@ -0,0 +1,35 @@ +import sys + +# couple of imports to see if packages are working +import numpy +import pkg_resources + + +def test_pytorch(): + import torch + + from torchvision.models import DenseNet + + model = DenseNet() + t = torch.randn(1, 3, 224, 224) + out = model(t) + assert out.shape[1] == 1000 + + +def _check_package(name, pyname=None): + "Checks if a Python package can be `require()`'d" + + pyname = pyname or name + print(f"Checking Python setuptools integrity for {name} (pyname: {pyname})") + pkg_resources.require(pyname) + + +def test_setuptools_integrity(): + + _check_package("pytorch", "torch") + _check_package("torchvision") + + +# test if pytorch installation is sane +test_pytorch() +test_setuptools_integrity()