#!/usr/bin/env python3 """ This script should use a very simple, functional programming style. Avoid Jinja macros in favor of native Python functions. Don't go overboard on code generation; use Python only to generate content that can't be easily declared statically using CircleCI's YAML API. Data declarations (e.g. the nested loops for defining the configuration matrix) should be at the top of the file for easy updating. See this comment for design rationale: https://github.com/pytorch/vision/pull/1321#issuecomment-531033978 """ import jinja2 import yaml import os.path PYTHON_VERSIONS = ["3.6", "3.7", "3.8"] def build_workflows(prefix='', filter_branch=None, upload=False, indentation=6, windows_latest_only=False): w = [] for btype in ["wheel", "conda"]: for os_type in ["linux", "macos", "win"]: python_versions = PYTHON_VERSIONS cu_versions_dict = {"linux": ["cpu", "cu92", "cu101", "cu102", "cu110"], "win": ["cpu", "cu101", "cu102", "cu110"], "macos": ["cpu"]} cu_versions = cu_versions_dict[os_type] for python_version in python_versions: for cu_version in cu_versions: for unicode in ([False, True] if btype == "wheel" and python_version == "2.7" else [False]): fb = filter_branch if windows_latest_only and os_type == "win" and filter_branch is None and \ (python_version != python_versions[-1] or (cu_version not in [cu_versions[0], cu_versions[-1]])): fb = "master" w += workflow_pair( btype, os_type, python_version, cu_version, unicode, prefix, upload, filter_branch=fb) return indent(indentation, w) def workflow_pair(btype, os_type, python_version, cu_version, unicode, prefix='', upload=False, *, filter_branch=None): w = [] unicode_suffix = "u" if unicode else "" base_workflow_name = f"{prefix}binary_{os_type}_{btype}_py{python_version}{unicode_suffix}_{cu_version}" w.append(generate_base_workflow( base_workflow_name, python_version, cu_version, unicode, os_type, btype, filter_branch=filter_branch)) if upload: w.append(generate_upload_workflow(base_workflow_name, os_type, btype, cu_version, filter_branch=filter_branch)) if filter_branch == 'nightly' and os_type in ['linux', 'win']: pydistro = 'pip' if btype == 'wheel' else 'conda' w.append(generate_smoketest_workflow(pydistro, base_workflow_name, filter_branch, python_version, os_type)) return w manylinux_images = { "cu92": "pytorch/manylinux-cuda92", "cu101": "pytorch/manylinux-cuda101", "cu102": "pytorch/manylinux-cuda102", "cu110": "pytorch/manylinux-cuda110", } def get_manylinux_image(cu_version): cu_suffix = "102" if cu_version.startswith('cu'): cu_suffix = cu_version[len('cu'):] return f"pytorch/manylinux-cuda{cu_suffix}" def generate_base_workflow(base_workflow_name, python_version, cu_version, unicode, os_type, btype, *, filter_branch=None): d = { "name": base_workflow_name, "python_version": python_version, "cu_version": cu_version, } if os_type != "win" and unicode: d["unicode_abi"] = '1' if os_type != "win": d["wheel_docker_image"] = get_manylinux_image(cu_version) if filter_branch is not None: d["filters"] = { "branches": { "only": filter_branch }, "tags": { # Using a raw string here to avoid having to escape # anything "only": r"/v[0-9]+(\.[0-9]+)*-rc[0-9]+/" } } w = f"binary_{os_type}_{btype}" return {w: d} def gen_filter_branch_tree(*branches): return {"branches": {"only": [b for b in branches]}} def generate_upload_workflow(base_workflow_name, os_type, btype, cu_version, *, filter_branch=None): d = { "name": f"{base_workflow_name}_upload", "context": "org-member", "requires": [base_workflow_name], } if btype == 'wheel': d["subfolder"] = "" if os_type == 'macos' else cu_version + "/" if filter_branch is not None: d["filters"] = { "branches": { "only": filter_branch }, "tags": { # Using a raw string here to avoid having to escape # anything "only": r"/v[0-9]+(\.[0-9]+)*-rc[0-9]+/" } } return {f"binary_{btype}_upload": d} def generate_smoketest_workflow(pydistro, base_workflow_name, filter_branch, python_version, os_type): required_build_suffix = "_upload" required_build_name = base_workflow_name + required_build_suffix smoke_suffix = f"smoke_test_{pydistro}" d = { "name": f"{base_workflow_name}_{smoke_suffix}", "requires": [required_build_name], "python_version": python_version, } if filter_branch: d["filters"] = gen_filter_branch_tree(filter_branch) return {"smoke_test_{os_type}_{pydistro}".format(os_type=os_type, pydistro=pydistro): d} def indent(indentation, data_list): return ("\n" + " " * indentation).join( yaml.dump(data_list, default_flow_style=False).splitlines()) def unittest_workflows(indentation=6): jobs = [] for os_type in ["linux", "windows", "macos"]: for device_type in ["cpu", "gpu"]: if os_type == "macos" and device_type == "gpu": continue for i, python_version in enumerate(PYTHON_VERSIONS): job = { "name": f"unittest_{os_type}_{device_type}_py{python_version}", "python_version": python_version, } if device_type == 'gpu': if python_version != "3.8": job['filters'] = gen_filter_branch_tree('master', 'nightly') job['cu_version'] = 'cu101' else: job['cu_version'] = 'cpu' jobs.append({f"unittest_{os_type}_{device_type}": job}) return indent(indentation, jobs) def cmake_workflows(indentation=6): jobs = [] python_version = '3.8' for os_type in ['linux', 'windows', 'macos']: # Skip OSX CUDA device_types = ['cpu', 'gpu'] if os_type != 'macos' else ['cpu'] for device in device_types: job = { 'name': f'cmake_{os_type}_{device}', 'python_version': python_version } job['cu_version'] = 'cu101' if device == 'gpu' else 'cpu' if device == 'gpu' and os_type == 'linux': job['wheel_docker_image'] = 'pytorch/manylinux-cuda101' jobs.append({f'cmake_{os_type}_{device}': job}) return indent(indentation, jobs) if __name__ == "__main__": d = os.path.dirname(__file__) env = jinja2.Environment( loader=jinja2.FileSystemLoader(d), lstrip_blocks=True, autoescape=False, ) with open(os.path.join(d, 'config.yml'), 'w') as f: f.write(env.get_template('config.yml.in').render( build_workflows=build_workflows, unittest_workflows=unittest_workflows, cmake_workflows=cmake_workflows, ))