2021-02-16 17:21:49 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
#
|
|
|
|
# Copyright (C) 2021 Red Hat, Inc.
|
|
|
|
# SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
|
|
|
|
import argparse
|
2021-03-12 17:55:08 +01:00
|
|
|
import os
|
2021-02-16 17:21:49 +01:00
|
|
|
import pathlib
|
2021-03-12 17:52:50 +01:00
|
|
|
import pty
|
2021-03-12 17:48:47 +01:00
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
import sys
|
2021-03-15 15:42:13 +01:00
|
|
|
import textwrap
|
2021-02-16 17:21:49 +01:00
|
|
|
|
2021-03-16 09:47:23 +01:00
|
|
|
import util
|
|
|
|
|
2021-02-16 17:21:49 +01:00
|
|
|
|
|
|
|
class Parser:
|
|
|
|
def __init__(self):
|
2021-03-12 17:55:08 +01:00
|
|
|
# Options that are common to all actions that use containers
|
|
|
|
containerparser = argparse.ArgumentParser(add_help=False)
|
|
|
|
containerparser.add_argument(
|
|
|
|
"target",
|
|
|
|
help="perform action on target OS",
|
|
|
|
)
|
|
|
|
containerparser.add_argument(
|
|
|
|
"--engine",
|
|
|
|
choices=["auto", "podman", "docker"],
|
|
|
|
default="auto",
|
|
|
|
help="container engine to use",
|
|
|
|
)
|
|
|
|
containerparser.add_argument(
|
|
|
|
"--login",
|
|
|
|
default=os.getlogin(), # exempt from syntax-check
|
|
|
|
help="login to use inside the container",
|
|
|
|
)
|
|
|
|
containerparser.add_argument(
|
|
|
|
"--image-prefix",
|
|
|
|
default="registry.gitlab.com/libvirt/libvirt/ci-",
|
|
|
|
help="use container images from non-default location",
|
|
|
|
)
|
|
|
|
containerparser.add_argument(
|
|
|
|
"--image-tag",
|
|
|
|
default=":latest",
|
|
|
|
help="use container images with non-default tags",
|
|
|
|
)
|
|
|
|
|
2021-03-12 18:00:52 +01:00
|
|
|
# Options that are common to all actions that call the
|
|
|
|
# project's build system
|
|
|
|
mesonparser = argparse.ArgumentParser(add_help=False)
|
|
|
|
mesonparser.add_argument(
|
|
|
|
"--meson-args",
|
|
|
|
default="",
|
|
|
|
help="additional arguments passed to meson "
|
|
|
|
"(eg --meson-args='-Dopt1=enabled -Dopt2=disabled')",
|
|
|
|
)
|
|
|
|
mesonparser.add_argument(
|
|
|
|
"--ninja-args",
|
|
|
|
default="",
|
|
|
|
help="additional arguments passed to ninja",
|
|
|
|
)
|
|
|
|
|
2021-03-12 17:48:47 +01:00
|
|
|
# Options that are common to all actions that use lcitool
|
|
|
|
lcitoolparser = argparse.ArgumentParser(add_help=False)
|
|
|
|
lcitoolparser.add_argument(
|
|
|
|
"--lcitool",
|
|
|
|
metavar="PATH",
|
|
|
|
default="lcitool",
|
|
|
|
help="path to lcitool binary",
|
|
|
|
)
|
|
|
|
|
2021-03-16 09:47:23 +01:00
|
|
|
# Options that are common to actions communicating with a GitLab
|
|
|
|
# instance
|
|
|
|
gitlabparser = argparse.ArgumentParser(add_help=False)
|
|
|
|
gitlabparser.add_argument(
|
|
|
|
"--namespace",
|
|
|
|
default="libvirt/libvirt",
|
|
|
|
help="GitLab project namespace"
|
|
|
|
)
|
|
|
|
gitlabparser.add_argument(
|
|
|
|
"--gitlab-uri",
|
|
|
|
default="https://gitlab.com",
|
|
|
|
help="base GitLab URI"
|
|
|
|
)
|
|
|
|
|
2021-02-16 17:21:49 +01:00
|
|
|
# Main parser
|
2021-03-18 08:34:18 +01:00
|
|
|
self._parser = argparse.ArgumentParser()
|
|
|
|
subparsers = self._parser.add_subparsers(
|
2021-02-16 17:21:49 +01:00
|
|
|
dest="action",
|
|
|
|
metavar="ACTION",
|
|
|
|
)
|
|
|
|
subparsers.required = True
|
|
|
|
|
2021-03-12 18:00:52 +01:00
|
|
|
# build action
|
|
|
|
buildparser = subparsers.add_parser(
|
|
|
|
"build",
|
|
|
|
help="run a build in a container",
|
|
|
|
parents=[containerparser, mesonparser],
|
2021-03-16 15:44:06 +01:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
|
2021-03-12 18:00:52 +01:00
|
|
|
)
|
2021-03-18 08:34:18 +01:00
|
|
|
buildparser.set_defaults(func=Application._action_build)
|
2021-03-12 18:00:52 +01:00
|
|
|
|
2021-03-12 18:01:43 +01:00
|
|
|
# test action
|
|
|
|
testparser = subparsers.add_parser(
|
|
|
|
"test",
|
|
|
|
help="run a build in a container (including tests)",
|
|
|
|
parents=[containerparser, mesonparser],
|
2021-03-16 15:44:06 +01:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
|
2021-03-12 18:01:43 +01:00
|
|
|
)
|
2021-03-18 08:34:18 +01:00
|
|
|
testparser.set_defaults(func=Application._action_test)
|
2021-03-12 18:01:43 +01:00
|
|
|
|
2021-03-12 17:55:08 +01:00
|
|
|
# shell action
|
|
|
|
shellparser = subparsers.add_parser(
|
|
|
|
"shell",
|
|
|
|
help="start a shell in a container",
|
|
|
|
parents=[containerparser],
|
2021-03-16 15:44:06 +01:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
|
2021-03-12 17:55:08 +01:00
|
|
|
)
|
2021-03-18 08:34:18 +01:00
|
|
|
shellparser.set_defaults(func=Application._action_shell)
|
2021-03-12 17:55:08 +01:00
|
|
|
|
2021-03-12 17:52:50 +01:00
|
|
|
# list-images action
|
|
|
|
listimagesparser = subparsers.add_parser(
|
|
|
|
"list-images",
|
|
|
|
help="list known container images",
|
2021-03-16 09:47:23 +01:00
|
|
|
parents=[gitlabparser],
|
2021-03-16 15:44:06 +01:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
|
2021-03-12 17:52:50 +01:00
|
|
|
)
|
2021-03-18 08:34:18 +01:00
|
|
|
listimagesparser.set_defaults(func=Application._action_list_images)
|
2021-03-12 17:52:50 +01:00
|
|
|
|
2021-03-12 17:48:47 +01:00
|
|
|
# refresh action
|
|
|
|
refreshparser = subparsers.add_parser(
|
|
|
|
"refresh",
|
|
|
|
help="refresh data generated with lcitool",
|
2021-03-15 15:42:13 +01:00
|
|
|
parents=[lcitoolparser, gitlabparser],
|
2021-03-16 15:44:06 +01:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
|
2021-03-12 17:48:47 +01:00
|
|
|
)
|
2021-03-16 18:00:41 +01:00
|
|
|
refreshparser.add_argument(
|
|
|
|
"--quiet",
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
|
|
|
help="refresh data silently"
|
|
|
|
)
|
2021-03-15 15:42:13 +01:00
|
|
|
refreshparser.add_argument(
|
|
|
|
"--check-stale",
|
|
|
|
action="store",
|
|
|
|
choices=["yes", "no"],
|
|
|
|
default="yes",
|
|
|
|
help="check for existence of stale images on the GitLab instance"
|
|
|
|
)
|
2021-03-18 08:34:18 +01:00
|
|
|
refreshparser.set_defaults(func=Application._action_refresh)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-02-16 17:21:49 +01:00
|
|
|
def parse(self):
|
2021-03-18 08:34:18 +01:00
|
|
|
return self._parser.parse_args()
|
2021-02-16 17:21:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Application:
|
|
|
|
def __init__(self):
|
2021-03-18 08:34:18 +01:00
|
|
|
self._basedir = pathlib.Path(__file__).resolve().parent
|
|
|
|
self._args = Parser().parse()
|
2021-02-16 17:21:49 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
if self._args.action == "refresh":
|
|
|
|
if not shutil.which(self._args.lcitool):
|
2021-03-12 17:48:47 +01:00
|
|
|
sys.exit("error: 'lcitool' not installed")
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _make_run(self, target):
|
2021-03-12 17:52:50 +01:00
|
|
|
args = [
|
|
|
|
"-C",
|
2021-03-18 08:34:18 +01:00
|
|
|
self._basedir,
|
2021-03-12 17:52:50 +01:00
|
|
|
target,
|
|
|
|
]
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
if self._args.action in ["build", "test", "shell"]:
|
2021-03-12 17:55:08 +01:00
|
|
|
args.extend([
|
2021-03-18 08:34:18 +01:00
|
|
|
f"CI_ENGINE={self._args.engine}",
|
|
|
|
f"CI_USER_LOGIN={self._args.login}",
|
|
|
|
f"CI_IMAGE_PREFIX={self._args.image_prefix}",
|
|
|
|
f"CI_IMAGE_TAG={self._args.image_tag}",
|
2021-03-12 17:55:08 +01:00
|
|
|
])
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
if self._args.action in ["build", "test"]:
|
2021-03-12 18:00:52 +01:00
|
|
|
args.extend([
|
2021-03-18 08:34:18 +01:00
|
|
|
f"CI_MESON_ARGS={self._args.meson_args}",
|
|
|
|
f"CI_NINJA_ARGS={self._args.ninja_args}",
|
2021-03-12 18:00:52 +01:00
|
|
|
])
|
|
|
|
|
2021-03-12 17:52:50 +01:00
|
|
|
if pty.spawn(["make"] + args) != 0:
|
|
|
|
sys.exit("error: 'make' failed")
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _lcitool_run(self, args):
|
|
|
|
output = subprocess.check_output([self._args.lcitool] + args)
|
2021-03-12 17:48:47 +01:00
|
|
|
return output.decode("utf-8")
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
def _lcitool_get_targets(self):
|
|
|
|
output = self._lcitool_run(["targets"])
|
2021-03-12 17:48:47 +01:00
|
|
|
return output.splitlines()
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
def _generate_dockerfile(self, target, cross=None):
|
|
|
|
args = ["dockerfile", target, "libvirt"]
|
2021-03-18 08:34:18 +01:00
|
|
|
outdir = self._basedir.joinpath("containers")
|
2021-07-22 15:58:28 +02:00
|
|
|
outfile = f"{target}.Dockerfile"
|
2021-03-12 17:48:47 +01:00
|
|
|
|
|
|
|
if cross:
|
|
|
|
args.extend(["--cross", cross])
|
2021-07-22 15:58:28 +02:00
|
|
|
outfile = f"{target}-cross-{cross}.Dockerfile"
|
2021-03-12 17:48:47 +01:00
|
|
|
|
|
|
|
outpath = outdir.joinpath(outfile)
|
2021-03-18 08:34:18 +01:00
|
|
|
if not self._args.quiet:
|
2021-03-16 18:00:41 +01:00
|
|
|
print(outpath)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
output = self._lcitool_run(args)
|
2021-03-12 17:48:47 +01:00
|
|
|
with open(outpath, "w") as f:
|
|
|
|
f.write(output)
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
def _generate_vars(self, target):
|
|
|
|
args = ["variables", target, "libvirt"]
|
2021-03-18 08:34:18 +01:00
|
|
|
outdir = self._basedir.joinpath("cirrus")
|
2021-07-22 15:58:28 +02:00
|
|
|
outfile = f"{target}.vars"
|
2021-03-12 17:48:47 +01:00
|
|
|
|
|
|
|
outpath = outdir.joinpath(outfile)
|
2021-03-18 08:34:18 +01:00
|
|
|
if not self._args.quiet:
|
2021-03-16 18:00:41 +01:00
|
|
|
print(outpath)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
output = self._lcitool_run(args)
|
2021-03-12 17:48:47 +01:00
|
|
|
with open(outpath, "w") as f:
|
|
|
|
f.write(output)
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _refresh_containers(self):
|
2021-03-12 17:48:47 +01:00
|
|
|
debian_cross = [
|
|
|
|
"aarch64",
|
|
|
|
"armv6l",
|
|
|
|
"armv7l",
|
|
|
|
"i686",
|
|
|
|
"mips",
|
|
|
|
"mips64el",
|
|
|
|
"mipsel",
|
|
|
|
"ppc64le",
|
|
|
|
"s390x",
|
|
|
|
]
|
|
|
|
fedora_cross = [
|
|
|
|
"mingw32",
|
|
|
|
"mingw64",
|
|
|
|
]
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
for target in self._lcitool_get_targets():
|
|
|
|
if target.startswith("freebsd-") or target.startswith("macos-"):
|
2021-03-12 17:48:47 +01:00
|
|
|
continue
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
self._generate_dockerfile(target)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
if target == "fedora-rawhide":
|
2021-03-12 17:48:47 +01:00
|
|
|
for cross in fedora_cross:
|
2021-07-22 15:58:28 +02:00
|
|
|
self._generate_dockerfile(target, cross)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
if target.startswith("debian-"):
|
2021-03-12 17:48:47 +01:00
|
|
|
for cross in debian_cross:
|
2021-07-22 15:58:28 +02:00
|
|
|
if target == "debian-sid" and cross == "mips":
|
2021-03-12 17:48:47 +01:00
|
|
|
continue
|
2021-07-22 15:58:28 +02:00
|
|
|
self._generate_dockerfile(target, cross)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _refresh_cirrus(self):
|
2021-07-22 15:58:28 +02:00
|
|
|
for target in self._lcitool_get_targets():
|
|
|
|
if not (target.startswith("freebsd-") or target.startswith("macos-")):
|
2021-03-12 17:48:47 +01:00
|
|
|
continue
|
|
|
|
|
2021-07-22 15:58:28 +02:00
|
|
|
self._generate_vars(target)
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _check_stale_images(self):
|
|
|
|
namespace = self._args.namespace
|
|
|
|
gitlab_uri = self._args.gitlab_uri
|
2021-03-15 15:42:13 +01:00
|
|
|
registry_uri = util.get_registry_uri(namespace, gitlab_uri)
|
2021-07-22 15:58:28 +02:00
|
|
|
lcitool_targets = self._lcitool_get_targets()
|
2021-03-15 15:42:13 +01:00
|
|
|
|
|
|
|
stale_images = util.get_registry_stale_images(registry_uri,
|
2021-07-22 15:58:28 +02:00
|
|
|
lcitool_targets)
|
2021-03-15 15:42:13 +01:00
|
|
|
if stale_images:
|
|
|
|
spacing = "\n" + 4 * " "
|
|
|
|
stale_fmt = [f"{k} (ID: {v})" for k, v in stale_images.items()]
|
|
|
|
stale_details = spacing.join(stale_fmt)
|
|
|
|
stale_ids = ' '.join([str(id) for id in stale_images.values()])
|
|
|
|
registry_uri = util.get_registry_uri(namespace, gitlab_uri)
|
|
|
|
|
|
|
|
msg = textwrap.dedent(f"""
|
|
|
|
The following images are stale and can be purged from the registry:
|
|
|
|
|
|
|
|
STALE_DETAILS
|
|
|
|
|
|
|
|
You can delete the images listed above using this shell snippet:
|
|
|
|
|
|
|
|
$ for image_id in {stale_ids}; do
|
|
|
|
curl --request DELETE --header "PRIVATE-TOKEN: <access_token>" \\
|
|
|
|
{registry_uri}/$image_id;
|
|
|
|
done
|
|
|
|
|
|
|
|
You can generate a personal access token here:
|
|
|
|
|
|
|
|
{gitlab_uri}/-/profile/personal_access_tokens
|
|
|
|
""")
|
|
|
|
print(msg.replace("STALE_DETAILS", stale_details))
|
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _action_build(self):
|
|
|
|
self._make_run(f"ci-build@{self._args.target}")
|
2021-03-12 18:00:52 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _action_test(self):
|
|
|
|
self._make_run(f"ci-test@{self._args.target}")
|
2021-03-12 18:01:43 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _action_shell(self):
|
|
|
|
self._make_run(f"ci-shell@{self._args.target}")
|
2021-03-12 17:55:08 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _action_list_images(self):
|
|
|
|
registry_uri = util.get_registry_uri(self._args.namespace,
|
|
|
|
self._args.gitlab_uri)
|
2021-03-16 09:47:23 +01:00
|
|
|
images = util.get_registry_images(registry_uri)
|
|
|
|
|
|
|
|
# skip the "ci-" prefix each of our container images' name has
|
|
|
|
name_prefix = "ci-"
|
|
|
|
names = [i["name"][len(name_prefix):] for i in images]
|
|
|
|
names.sort()
|
|
|
|
|
|
|
|
native = [name for name in names if "-cross-" not in name]
|
|
|
|
cross = [name for name in names if "-cross-" in name]
|
|
|
|
|
|
|
|
spacing = 4 * " "
|
|
|
|
print("Available x86 container images:\n")
|
|
|
|
print(spacing + ("\n" + spacing).join(native))
|
|
|
|
|
|
|
|
if cross:
|
|
|
|
print()
|
|
|
|
print("Available cross-compiler container images:\n")
|
|
|
|
print(spacing + ("\n" + spacing).join(cross))
|
2021-03-12 17:52:50 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
def _action_refresh(self):
|
|
|
|
self._refresh_containers()
|
|
|
|
self._refresh_cirrus()
|
2021-03-12 17:48:47 +01:00
|
|
|
|
2021-03-18 08:34:18 +01:00
|
|
|
if self._args.check_stale == "yes" and not self._args.quiet:
|
|
|
|
self._check_stale_images()
|
2021-03-15 15:42:13 +01:00
|
|
|
|
2021-02-16 17:21:49 +01:00
|
|
|
def run(self):
|
2021-03-18 08:34:18 +01:00
|
|
|
self._args.func(self)
|
2021-02-16 17:21:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
Application().run()
|