-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathbuild.py
executable file
·166 lines (143 loc) · 4.67 KB
/
build.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
#! /usr/bin/env python3
# This script uses only the standardlib because that is what
# is available on in the context of the CI/CD platform.
import os
import argparse
def get_image_name(repo, tag, target):
if target == "prod":
return "{}:{}".format(repo, tag)
else:
return "{}:{}-{}".format(repo, tag, target)
def parse_image_name(image_name):
repo, tag = image_name.rsplit(":", 1)
if tag.endswith("-dev"):
tag = tag[:-4]
target = "dev"
else:
target = "prod"
return repo, tag, target
def get_context_path_from_tag(tag):
version, directory = tag.split("-", 1)
if directory.endswith("-dev"):
directory = directory[:-4]
return directory
def get_build_command(repo, tag, target):
return [
"docker",
"build",
"-t",
get_image_name(repo, tag, target),
"--build-arg",
"TARGET={}".format(target),
"--no-cache",
get_context_path_from_tag(tag=tag),
]
def get_test_command(repo, tag, target):
"""
Builds and returns a command to test the resulting image.
The tests execute the following operations:
* `pip-reqs` `compile` and `resolve` on an arbitrary platform (it's not an
issue if the platform does not match the platform of the image, as the
resulting requirements.urls file will never be used);
* `pip install` a series of packages known to stress the build phase (we
force them to be build from source).
Note that the `pip-reqs` and the `pip install` steps are completely
independent on purpose (the installation step does not use the resolved
requirements from `pip-reqs`). We are just smoke testing `pip-reqs` on the
first step, and making sure that we are able to use the dev image to build
binary wheels in the second step.
"""
with open(".artifacts/test/requirements.in", "w") as fh:
fh.write("django\n")
packages = [
"psycopg2",
"cryptography",
"pybind11",
"numpy",
"scipy",
"pillow",
"lxml",
"pyyaml",
"easy_thumbnails[svg]",
]
accept_wheels = [
"importlib-metadata",
"zipp",
"freetype-py",
]
pip_command = "pip install --no-binary :all: --only-binary {} {}".format(
",".join(accept_wheels),
" ".join(packages),
)
return [
"docker",
"run",
"-v"
"{}/.artifacts/test:/app:rw".format(
os.path.abspath(os.path.dirname(__file__))
),
"-e",
"WHEELSPROXY_URL=https://wheels.aldryn.net/v1/pypi/buster-py39/",
"-e",
"NPY_NUM_BUILD_JOBS={}".format(os.cpu_count()),
get_image_name(repo, tag, target),
"sh",
"-c",
"pip-reqs compile && pip-reqs resolve && {}".format(pip_command),
]
def main():
parser = argparse.ArgumentParser(
formatter_class=argparse.ArgumentDefaultsHelpFormatter
)
parser.add_argument("operation", help='The operation: "build" or "test"')
parser.add_argument(
"--repo",
default=os.environ.get("DOCKER_REPO", "divio/base"),
help="The repository name",
)
parser.add_argument(
"--tag",
default=os.environ.get("DOCKER_TAG", ""),
help='A tag name like "4.42-py3.6-alpine". Do not include the "-dev" '
"suffix, instead set the --target option.",
)
parser.add_argument(
"--target",
default=os.environ.get("TARGET", "prod"),
help="The build target (dev or prod).",
)
args = parser.parse_args()
operation = args.operation
image_name = os.environ.get("IMAGE_NAME")
if image_name:
# When dockerhub builds IMAGE_NAME will be set and will override any
# other options.
repo, tag, target = parse_image_name(image_name=image_name)
else:
# For local building we can use these env vars.
repo = str(args.repo)
tag = str(args.tag)
target = str(args.target)
if not (repo and tag and target):
print("Missing parameters!")
exit(code=1)
if tag.endswith("-dev"):
print(
"Do not include the -dev suffix in the tag. "
"Set `--target` instead."
)
exit(code=1)
if operation == "build":
command = get_build_command(
repo=repo, tag=tag, target=target
)
elif operation == "test":
command = get_test_command(repo=repo, tag=tag, target=target)
print("repo: {}".format(repo))
print("tag: {}".format(tag))
print("target: {}".format(target))
print("command:")
print(" ".join(command))
os.execvp(command[0], command)
if __name__ == "__main__":
main()