2021-02-16 19:21:49 +03:00
#!/usr/bin/env python3
#
# Copyright (C) 2021 Red Hat, Inc.
# SPDX-License-Identifier: LGPL-2.1-or-later
import argparse
2021-03-12 19:55:08 +03:00
import os
2021-02-16 19:21:49 +03:00
import pathlib
2021-03-12 19:48:47 +03:00
import subprocess
import sys
2021-03-15 17:42:13 +03:00
import textwrap
2021-02-16 19:21:49 +03:00
2023-08-24 14:21:12 +03:00
from pathlib import Path
from tempfile import TemporaryDirectory
2021-03-16 11:47:23 +03:00
import util
2021-02-16 19:21:49 +03:00
2023-08-24 11:28:55 +03:00
def required_deps(*deps):
module2pkg = {
"git": "GitPython"
}
def inner_decorator(func):
def wrapped(*args, **kwargs):
cmd = func.__name__[len('_action_'):]
for dep in deps:
try:
import importlib
importlib.import_module(dep)
except ImportError:
pkg = module2pkg[dep]
msg = f"'{pkg}' not found (required by the '{cmd}' command)"
print(msg, file=sys.stderr)
sys.exit(1)
func(*args, **kwargs)
return wrapped
return inner_decorator
2021-02-16 19:21:49 +03:00
class Parser:
def __init__(self):
2021-03-12 19:55:08 +03: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",
2023-08-24 11:32:24 +03:00
default="latest",
2021-03-12 19:55:08 +03:00
help="use container images with non-default tags",
)
2023-08-24 11:33:43 +03:00
containerparser.add_argument(
"--lcitool-path",
dest="lcitool",
default="lcitool",
help="path to lcitool (default: $PATH)",
)
2021-03-12 19:55:08 +03:00
2021-03-16 11:47:23 +03: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 19:21:49 +03:00
# Main parser
2021-03-18 10:34:18 +03:00
self._parser = argparse.ArgumentParser()
subparsers = self._parser.add_subparsers(
2021-02-16 19:21:49 +03:00
dest="action",
metavar="ACTION",
)
subparsers.required = True
2023-08-24 12:07:09 +03:00
jobparser = subparsers.add_parser(
"run",
help="Run a GitLab CI job or 'shell' in a local environment",
parents=[containerparser],
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
jobparser.add_argument(
"--job",
choices=["build", "codestyle", "potfile", "rpmbuild",
"shell", "test", "website"],
default="build",
help="Run a GitLab CI job or 'shell' in a local environment",
)
jobparser.set_defaults(func=Application._action_run)
2021-03-12 19:52:50 +03:00
# list-images action
listimagesparser = subparsers.add_parser(
"list-images",
help="list known container images",
2021-03-16 11:47:23 +03:00
parents=[gitlabparser],
2021-03-16 17:44:06 +03:00
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
2021-03-12 19:52:50 +03:00
)
2021-03-18 10:34:18 +03:00
listimagesparser.set_defaults(func=Application._action_list_images)
2021-03-12 19:52:50 +03:00
2021-09-09 16:20:44 +03:00
# check_stale action
check_staleparser = subparsers.add_parser(
"check-stale",
help="check for existence of stale images on the GitLab instance",
parents=[gitlabparser],
2021-03-16 17:44:06 +03:00
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
2021-03-12 19:48:47 +03:00
)
2021-09-09 16:20:44 +03:00
check_staleparser.set_defaults(func=Application._action_check_stale)
2021-03-12 19:48:47 +03:00
2021-02-16 19:21:49 +03:00
def parse(self):
2021-03-18 10:34:18 +03:00
return self._parser.parse_args()
2021-02-16 19:21:49 +03:00
class Application:
2023-08-24 11:24:12 +03:00
@property
def repo(self):
if self._repo is None:
from git import Repo
self._repo = Repo(search_parent_directories=True)
return self._repo
2021-02-16 19:21:49 +03:00
def __init__(self):
2021-03-18 10:34:18 +03:00
self._basedir = pathlib.Path(__file__).resolve().parent
self._args = Parser().parse()
2023-08-24 11:24:12 +03:00
self._repo = None
2021-02-16 19:21:49 +03:00
2023-08-24 12:13:59 +03:00
@staticmethod
def _prepare_repo_copy(repo, dest):
return repo.clone(dest, local=True)
2021-03-18 10:34:18 +03:00
def _lcitool_run(self, args):
2023-08-24 14:21:12 +03:00
positional_args = ["container"]
opts = ["--user", self._args.login]
tmpdir = TemporaryDirectory(prefix="scratch",
dir=Path(self.repo.working_dir, "ci"))
repo_dest_path = Path(tmpdir.name, "libvirt.git").as_posix()
repo_clone = self._prepare_repo_copy(self.repo, repo_dest_path)
opts.extend(["--workload-dir", repo_clone.working_dir])
if self._args.job == "shell":
positional_args.append("shell")
else:
job2func = {
"test": "run_test",
"build": "run_build",
"codestyle": "run_codestyle",
"potfile": "run_potfile",
"rpmbuild": "run_rpmbuild",
"website": "run_website_build",
}
if self._args.engine != "auto":
positional_args.extend(["--engine", self._args.engine])
with open(Path(tmpdir.name, "script"), "w") as f:
script_path = f.name
contents = textwrap.dedent(f"""\
#!/bin/sh
cd datadir
. ci/jobs.sh
{job2func[self._args.job]}
""")
f.write(contents)
positional_args.append("run")
opts.extend(["--script", script_path])
opts.append(f"{self._args.image_prefix}{self._args.target}:{self._args.image_tag}")
proc = None
try:
proc = subprocess.run([self._args.lcitool] + positional_args + opts)
except KeyboardInterrupt:
sys.exit(1)
finally:
# this will take care of the generated script file above as well
tmpdir.cleanup()
return proc.returncode
2021-03-12 19:48:47 +03:00
2021-03-18 10:34:18 +03:00
def _check_stale_images(self):
namespace = self._args.namespace
gitlab_uri = self._args.gitlab_uri
2021-03-15 17:42:13 +03:00
registry_uri = util.get_registry_uri(namespace, gitlab_uri)
2021-09-09 16:20:44 +03:00
stale_images = util.get_registry_stale_images(registry_uri, self._basedir)
2021-03-15 17:42:13 +03: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))
2023-08-24 16:21:44 +03:00
@required_deps("git")
def _action_run(self):
return self._lcitool_run(self._args.job)
2021-03-18 10:34:18 +03:00
def _action_list_images(self):
registry_uri = util.get_registry_uri(self._args.namespace,
self._args.gitlab_uri)
2021-03-16 11:47:23 +03: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 19:52:50 +03:00
2021-09-09 16:20:44 +03:00
def _action_check_stale(self):
self._check_stale_images()
2021-03-15 17:42:13 +03:00
2021-02-16 19:21:49 +03:00
def run(self):
2021-03-18 10:34:18 +03:00
self._args.func(self)
2021-02-16 19:21:49 +03:00
if __name__ == "__main__":
Application().run()