mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 14:59:49 +00:00
176 lines
5.8 KiB
Python
176 lines
5.8 KiB
Python
"""
|
|
This script reads the compact workflows.yml file, and and generates files in
|
|
.github/workflows/ suitable for the limited expressivity of GitHub's workflow
|
|
definition language.
|
|
|
|
The point is that we had/have a lot of duplications between files in
|
|
.github/workflows/, so we use this script to make it easier to update them
|
|
and keep them in sync.
|
|
"""
|
|
|
|
import enum
|
|
import pathlib
|
|
|
|
import yaml
|
|
|
|
ROOT_PATH = pathlib.Path(__file__).parent
|
|
DEFINITION_PATH = ROOT_PATH / "workflows.yml"
|
|
GH_WORKFLOW_DIR = ROOT_PATH / ".github" / "workflows"
|
|
|
|
|
|
class script:
|
|
def __init__(self, *lines):
|
|
self.data = "\n".join(lines)
|
|
|
|
|
|
def script_representer(dumper, data: script):
|
|
return dumper.represent_scalar("tag:yaml.org,2002:str", data.data, style="|")
|
|
|
|
|
|
class Dumper(yaml.Dumper):
|
|
pass
|
|
|
|
|
|
Dumper.add_representer(script, script_representer)
|
|
|
|
|
|
class VersionFlavor(enum.Enum):
|
|
STABLE = "stable"
|
|
"""A statically defined version, that we already tested irctest on.
|
|
This is ran on PRs and master, because failure guarantees it's a bug in
|
|
the new irctest commit/PR."""
|
|
RELEASE = "release"
|
|
"""The last release of the project. This should usually pass.
|
|
We don't currently use this."""
|
|
DEVEL = "devel"
|
|
"""The last commit of the project. This allows us to catch bugs in other
|
|
software early in their development process."""
|
|
DEVEL_RELEASE = "devel_release"
|
|
"""Ditto, but if the project uses a specific branch for their current
|
|
release series, it uses that branch instead"""
|
|
|
|
|
|
def generate_workflow(config: dict, software_id: str, version_flavor: VersionFlavor):
|
|
software_config = config["software"][software_id]
|
|
name = software_config["name"]
|
|
prefix = software_config.get("prefix", "~/.local")
|
|
|
|
if "install_steps" in software_config:
|
|
path = "placeholder" # TODO: remove this
|
|
install_steps = software_config["install_steps"][version_flavor.value]
|
|
if install_steps is None:
|
|
return
|
|
else:
|
|
ref = software_config["refs"][version_flavor.value]
|
|
if ref is None:
|
|
return
|
|
path = software_config["path"]
|
|
install_steps = [
|
|
{
|
|
"name": f"Checkout {name}",
|
|
"uses": "actions/checkout@v2",
|
|
"with": {
|
|
"repository": software_config["repository"],
|
|
"ref": ref,
|
|
"path": path,
|
|
},
|
|
},
|
|
{
|
|
"name": f"Build {name}",
|
|
"run": script(software_config["build_script"]),
|
|
},
|
|
]
|
|
|
|
on: dict
|
|
if version_flavor == VersionFlavor.STABLE:
|
|
on = {"push": None, "pull_request": None}
|
|
else:
|
|
# Run every saturday and sunday 8:51 UTC, and every day at 17:51
|
|
# (minute choosen at random, hours and days is so that I'm available
|
|
# to fix bugs it detects)
|
|
on = {
|
|
"schedule": [
|
|
{"cron": "51 8 * * 6"},
|
|
{"cron": "51 8 * * 0"},
|
|
{"cron": "51 17 * * *"},
|
|
],
|
|
"workflow_dispatch": None,
|
|
}
|
|
|
|
env = software_config.get("env", {}).get(version_flavor.value, "")
|
|
if env:
|
|
env += " "
|
|
|
|
workflow = {
|
|
"name": f"irctest with {name} ({version_flavor.value})",
|
|
"on": on,
|
|
"jobs": {
|
|
"build": {
|
|
"runs-on": "ubuntu-latest",
|
|
"steps": [
|
|
{"uses": "actions/checkout@v2"},
|
|
{
|
|
"name": "Set up Python 3.7", # for irctest itself
|
|
"uses": "actions/setup-python@v2",
|
|
"with": {"python-version": 3.7},
|
|
},
|
|
*software_config.get("pre_deps", []),
|
|
{
|
|
"name": "Cache dependencies",
|
|
"uses": "actions/cache@v2",
|
|
"with": {
|
|
"path": script("~/.cache", f"$GITHUB_WORKSPACE/{path}"),
|
|
"key": "${{ runner.os }}-" + software_id,
|
|
},
|
|
},
|
|
{
|
|
"name": "Install dependencies",
|
|
"run": script(
|
|
"sudo apt-get install atheme-services",
|
|
"python -m pip install --upgrade pip",
|
|
"pip install pytest -r requirements.txt",
|
|
*(
|
|
software_config["extra_deps"]
|
|
if "extra_deps" in software_config
|
|
else []
|
|
),
|
|
),
|
|
},
|
|
*install_steps,
|
|
{
|
|
"name": "Test with pytest",
|
|
"run": f"PATH={prefix}/bin:$PATH {env}make {software_id}",
|
|
},
|
|
],
|
|
}
|
|
},
|
|
}
|
|
|
|
if version_flavor == VersionFlavor.STABLE:
|
|
workflow_filename = GH_WORKFLOW_DIR / f"{software_id}.yml"
|
|
else:
|
|
workflow_filename = (
|
|
GH_WORKFLOW_DIR / f"{software_id}_{version_flavor.value}.yml"
|
|
)
|
|
|
|
with open(workflow_filename, "wt") as fd:
|
|
fd.write("# This file was auto-generated by make_workflows.py.\n")
|
|
fd.write("# Do not edit it manually, modifications will be lost.\n\n")
|
|
fd.write(yaml.dump(workflow, Dumper=Dumper))
|
|
|
|
|
|
def main():
|
|
with open(DEFINITION_PATH) as fd:
|
|
config = yaml.load(fd, Loader=yaml.Loader)
|
|
|
|
for software_id in config["software"]:
|
|
generate_workflow(config, software_id, version_flavor=VersionFlavor.STABLE)
|
|
generate_workflow(config, software_id, version_flavor=VersionFlavor.DEVEL)
|
|
generate_workflow(
|
|
config, software_id, version_flavor=VersionFlavor.DEVEL_RELEASE
|
|
)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|