regenerate.py 4.78 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#!/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
"""
16
17

import jinja2
18
import yaml
19
20
import os.path

21

moto's avatar
moto committed
22
23
24
25
PYTHON_VERSIONS = ["3.6", "3.7", "3.8"]


def build_workflows(prefix='', upload=False, filter_branch=None, indentation=6):
26
27
    w = []
    for btype in ["wheel", "conda"]:
peterjc123's avatar
peterjc123 committed
28
        for os_type in ["linux", "macos", "windows"]:
moto's avatar
moto committed
29
30
            for python_version in PYTHON_VERSIONS:
                w += build_workflow_pair(btype, os_type, python_version, filter_branch, prefix, upload)
31
32
33
34

    return indent(indentation, w)


moto's avatar
moto committed
35
def build_workflow_pair(btype, os_type, python_version, filter_branch, prefix='', upload=False):
36
37

    w = []
38
    base_workflow_name = "{prefix}binary_{os_type}_{btype}_py{python_version}".format(
39
40
41
42
        prefix=prefix,
        os_type=os_type,
        btype=btype,
        python_version=python_version,
43
    )
44

45
    w.append(generate_base_workflow(base_workflow_name, python_version, filter_branch, os_type, btype))
46
47
48

    if upload:

49
        is_py3_linux = os_type in ['linux', "windows"] and not python_version.startswith("2.")
50

51
        w.append(generate_upload_workflow(base_workflow_name, filter_branch, btype))
52
53
54

        if filter_branch == 'nightly' and is_py3_linux:
            pydistro = 'pip' if btype == 'wheel' else 'conda'
55
            w.append(generate_smoketest_workflow(pydistro, base_workflow_name, filter_branch, python_version, os_type))
56
57
58
59

    return w


60
def generate_base_workflow(base_workflow_name, python_version, filter_branch, os_type, btype):
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89

    d = {
        "name": base_workflow_name,
        "python_version": python_version,
    }

    if filter_branch:
        d["filters"] = gen_filter_branch_tree(filter_branch)

    return {"binary_{os_type}_{btype}".format(os_type=os_type, btype=btype): d}


def gen_filter_branch_tree(branch_name):
    return {"branches": {"only": branch_name}}


def generate_upload_workflow(base_workflow_name, filter_branch, btype):
    d = {
        "name": "{base_workflow_name}_upload".format(base_workflow_name=base_workflow_name),
        "context": "org-member",
        "requires": [base_workflow_name],
    }

    if filter_branch:
        d["filters"] = gen_filter_branch_tree(filter_branch)

    return {"binary_{btype}_upload".format(btype=btype): d}


90
def generate_smoketest_workflow(pydistro, base_workflow_name, filter_branch, python_version, os_type):
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105

    required_build_suffix = "_upload"
    required_build_name = base_workflow_name + required_build_suffix

    smoke_suffix = "smoke_test_{pydistro}".format(pydistro=pydistro)
    d = {
        "name": "{base_workflow_name}_{smoke_suffix}".format(
            base_workflow_name=base_workflow_name, smoke_suffix=smoke_suffix),
        "requires": [required_build_name],
        "python_version": python_version,
    }

    if filter_branch:
        d["filters"] = gen_filter_branch_tree(filter_branch)

106
    return {"smoke_test_{os_type}_{pydistro}".format(os_type=os_type, pydistro=pydistro): d}
107
108
109
110
111
112


def indent(indentation, data_list):
    return ("\n" + " " * indentation).join(yaml.dump(data_list).splitlines())


moto's avatar
moto committed
113
def unittest_workflows(indentation=6):
moto's avatar
moto committed
114
    jobs = []
115
    for os_type in ["linux", "windows"]:
moto's avatar
moto committed
116
        for device_type in ["cpu", "gpu"]:
117
            for i, python_version in enumerate(PYTHON_VERSIONS):
moto's avatar
moto committed
118
119
                job = {
                    "name": f"unittest_{os_type}_{device_type}_py{python_version}",
moto's avatar
moto committed
120
121
                    "python_version": python_version,
                }
moto's avatar
moto committed
122
123
124
125

                if device_type == 'gpu':
                    job['filters'] = gen_filter_branch_tree('master')
                jobs.append({f"unittest_{os_type}_{device_type}": job})
126
127
128
129
130
131
132
133

                if i == 0 and os_type == "linux" and device_type == "cpu":
                    jobs.append({
                        f"stylecheck": {
                            "name": f"stylecheck_py{python_version}",
                            "python_version": python_version,
                        }
                    })
moto's avatar
moto committed
134
    return indent(indentation, jobs)
moto's avatar
moto committed
135
136


137
138
139
140
141
142
143
144
145
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:
moto's avatar
moto committed
146
147
148
149
        f.write(env.get_template('config.yml.in').render(
            build_workflows=build_workflows,
            unittest_workflows=unittest_workflows,
        ))