1
1
mirror of https://github.com/wez/wezterm.git synced 2025-01-01 18:22:13 +03:00
wezterm/ci/generate-workflows.py
2020-11-29 10:26:11 -08:00

620 lines
18 KiB
Python
Executable File

#!/usr/bin/env python3
import os
import sys
def yv(v):
if v is True:
return "true"
if v is False:
return "false"
if v is None:
return "nil"
if isinstance(v, str):
if "\n" in v:
spacer = " " * 12
return "|\n" + spacer + v.replace("\n", "\n" + spacer) + "\n"
return '"' + v + '"'
return v
class Step(object):
def render(self, f, env):
raise NotImplementedError(repr(self))
class RunStep(Step):
def __init__(self, name, run, shell="bash"):
self.name = name
self.run = run
self.shell = shell
def render(self, f, env):
f.write(f" - name: {yv(self.name)}\n")
if self.shell:
f.write(f" shell: {self.shell}\n")
run = self.run
if env:
for k, v in env.items():
if self.shell is "bash":
run = f"export {k}={v}\n{run}\n"
f.write(f" run: {yv(run)}\n")
class ActionStep(Step):
def __init__(self, name, action, params=None, env=None):
self.name = name
self.action = action
self.params = params
self.env = env
def render(self, f, env):
f.write(f" - name: {yv(self.name)}\n")
f.write(f" uses: {self.action}\n")
if self.params:
f.write(" with:\n")
for k, v in self.params.items():
f.write(f" {k}: {yv(v)}\n")
if self.env:
f.write(" env:\n")
for k, v in self.env.items():
f.write(f" {k}: {yv(v)}\n")
class CacheStep(ActionStep):
def __init__(self, name, path, key):
super().__init__(
name, action="actions/cache@v2", params={"path": path, "key": key}
)
class CheckoutStep(ActionStep):
def __init__(self, name="checkout repo"):
super().__init__(
name, action="actions/checkout@v2", params={"submodules": "recursive"}
)
class Job(object):
def __init__(self, runs_on, container=None, steps=None, env=None):
self.runs_on = runs_on
self.container = container
self.steps = steps
self.env = env
def render(self, f):
for s in self.steps:
s.render(f, self.env)
class Target(object):
def __init__(
self,
name=None,
os="ubuntu-latest",
container=None,
bootstrap_git=False,
rust_target=None,
continuous_only=False,
app_image=False,
):
if not name:
if container:
name = container
else:
name = os
self.name = name.replace(":", "")
self.os = os
self.container = container
self.bootstrap_git = bootstrap_git
self.rust_target = rust_target
self.continuous_only = continuous_only
self.app_image = app_image
def uses_yum(self):
if "fedora" in self.name:
return True
if "centos" in self.name:
return True
return False
def uses_apt(self):
if "ubuntu" in self.name:
return True
if "debian" in self.name:
return True
return False
def needs_sudo(self):
if not self.container and self.uses_apt():
return True
return False
def install_system_package(self, name):
installer = None
if self.uses_yum():
installer = "yum"
elif self.uses_apt():
installer = "apt-get"
else:
return []
if self.needs_sudo():
installer = f"sudo -n {installer}"
return [RunStep(f"Install {name}", f"{installer} install -y {name}")]
def install_curl(self):
if self.uses_yum() or (self.uses_apt() and self.container):
return self.install_system_package("curl")
return []
def install_git(self):
steps = []
if self.bootstrap_git:
GIT_VERS = "2.26.2"
steps.append(
CacheStep(
"Cache Git installation",
path="/usr/local/git",
key=f"{self.name}-git-{GIT_VERS}",
)
)
pre_reqs = ""
if self.uses_yum():
pre_reqs = "yum install -y wget curl-devel expat-devel gettext-devel openssl-devel zlib-devel gcc perl-ExtUtils-MakeMaker make"
elif self.uses_apt():
pre_reqs = "apt-get install -y wget libcurl4-openssl-dev libexpat-dev gettext libssl-dev libz-dev gcc libextutils-autoinstall-perl make"
steps.append(
RunStep(
name="Install Git from source",
shell="bash",
run=f"""
{pre_reqs}
if test ! -x /usr/local/git/bin/git ; then
cd /tmp
wget https://github.com/git/git/archive/v{GIT_VERS}.tar.gz
tar xzf v{GIT_VERS}.tar.gz
cd git-{GIT_VERS}
make prefix=/usr/local/git install
fi
ln -s /usr/local/git/bin/git /usr/local/bin/git
""",
)
)
else:
steps += self.install_system_package("git")
return steps
def install_rust(self, cache=True):
salt = "2"
key_prefix = f"{self.name}-{self.rust_target}-{salt}-${{{{ runner.os }}}}-${{{{ hashFiles('**/Cargo.lock') }}}}"
params = {
"profile": "minimal",
"toolchain": "stable",
"override": True,
"components": "rustfmt",
}
if self.rust_target:
params["target"] = self.rust_target
steps = [
ActionStep(
name="Install Rust",
action="actions-rs/toolchain@v1",
params=params,
env={"ACTIONS_ALLOW_UNSECURE_COMMANDS": "true"},
),
]
if cache:
steps += [
CacheStep(
name="Cache cargo",
path="~/.cargo/registry\n~/.cargo/git\ntarget",
key=f"{key_prefix}-cargo",
),
]
return steps
def install_system_deps(self):
if "win" in self.name:
return []
sudo = "sudo -n " if self.needs_sudo() else ""
return [RunStep(name="Install System Deps", run=f"{sudo}./get-deps")]
def check_formatting(self):
return [RunStep(name="Check formatting", run="cargo fmt --all -- --check")]
def build_all_release(self):
if "win" in self.name:
return [
RunStep(
name="Build (Release mode)",
shell="cmd",
run="""
PATH C:\\Strawberry\\perl\\bin;%PATH%
cargo build --all --release""",
)
]
return [RunStep(name="Build (Release mode)", run="cargo build --all --release")]
def test_all_release(self):
return [RunStep(name="Test (Release mode)", run="cargo test --all --release")]
def package(self):
steps = [RunStep("Package", "bash ci/deploy.sh")]
if self.app_image:
steps.append(RunStep("Source Tarball", "bash ci/source-archive.sh"))
steps.append(RunStep("Build AppImage", "bash ci/appimage.sh"))
return steps
def upload_artifact(self):
run = "mkdir pkg_\n"
if self.uses_yum():
run += "mv ~/rpmbuild/RPMS/*/*.rpm pkg_\n"
if "win" in self.name:
run += "mv *.zip *.exe pkg_\n"
if "mac" in self.name:
run += "mv *.zip pkg_\n"
if ("ubuntu" in self.name) or ("debian" in self.name):
run += "mv *.deb *.xz pkg_\n"
if self.app_image:
run += "mv *.AppImage *.zsync pkg_\n"
return [
RunStep("Move Package for artifact upload", run),
ActionStep(
"Upload artifact",
action="actions/upload-artifact@master",
params={"name": self.name, "path": "pkg_"},
),
]
def asset_patterns(self):
patterns = []
if self.uses_yum():
patterns += ["wezterm-*.rpm"]
elif "win" in self.name:
patterns += ["WezTerm-*.zip", "WezTerm-*.exe"]
elif "mac" in self.name:
patterns += ["WezTerm-*.zip"]
elif ("ubuntu" in self.name) or ("debian" in self.name):
patterns += ["wezterm-*.deb", "wezterm-*.xz", "wezterm-*.tar.gz"]
if self.app_image:
patterns.append("*.AppImage")
patterns.append("*.zsync")
return patterns
def upload_asset_nightly(self):
steps = []
if self.uses_yum():
steps.append(
RunStep(
"Move RPM",
f"mv ~/rpmbuild/RPMS/*/*.rpm wezterm-nightly-{self.name}.rpm",
)
)
patterns = self.asset_patterns()
return steps + [
ActionStep(
"Upload to Nightly Release",
action="wez/upload-release-assets@releases/v1",
params={
"files": ";".join(patterns),
"release-tag": "nightly",
"repo-token": "${{ secrets.GITHUB_TOKEN }}",
},
)
]
def upload_asset_tag(self):
steps = []
if self.uses_yum():
steps.append(RunStep("Move RPM", "mv ~/rpmbuild/RPMS/*/*.rpm ."))
patterns = self.asset_patterns()
return steps + [
ActionStep(
"Upload to Tagged Release",
action="softprops/action-gh-release@v1",
params={"files": "\n".join(patterns), "prerelease": True},
env={
"GITHUB_TOKEN": "${{ secrets.GITHUB_TOKEN }}",
},
)
]
def update_homebrew_tap(self):
steps = []
if "macos" in self.name:
steps += [
ActionStep(
"Checkout homebrew tap",
action="actions/checkout@v2",
params={
"repository": "wez/homebrew-wezterm",
"path": "homebrew-wezterm",
"token": "${{ secrets.GH_PAT }}",
},
),
RunStep(
"Update homebrew tap formula",
"cp wezterm.rb homebrew-wezterm/Formula/wezterm.rb",
),
ActionStep(
"Commit homebrew tap changes",
action="stefanzweifel/git-auto-commit-action@v4",
params={
"commit_message": "Automated update to match latest tag",
"repository": "homebrew-wezterm",
},
),
]
elif self.app_image:
steps += [
ActionStep(
"Checkout linuxbrew tap",
action="actions/checkout@v2",
params={
"repository": "wez/homebrew-wezterm-linuxbrew",
"path": "linuxbrew-wezterm",
"token": "${{ secrets.GH_PAT }}",
},
),
RunStep(
"Update linuxbrew tap formula",
"cp wezterm-linuxbrew.rb linuxbrew-wezterm/Formula/wezterm.rb",
),
ActionStep(
"Commit linuxbrew tap changes",
action="stefanzweifel/git-auto-commit-action@v4",
params={
"commit_message": "Automated update to match latest tag",
"repository": "linuxbrew-wezterm",
},
),
]
return steps
def update_tagged_aur(self):
steps = []
if self.app_image:
# The AppImage build step also expands the PKGBUILD template
steps += [
ActionStep(
"Update AUR",
action="KSXGitHub/github-actions-deploy-aur@master",
params={
"pkgname": "wezterm-bin",
"pkgbuild": "PKGBUILD",
"commit_username": "wez",
"commit_email": "wez@wezfurlong.org",
"ssh_private_key": "${{ secrets.AUR_SSH_PRIVATE_KEY }}",
"commit_message": "Automated update to match latest tag",
},
)
]
return steps
def global_env(self):
env = {}
if "macos" in self.name:
env["MACOSX_DEPLOYMENT_TARGET"] = "10.9"
return env
def prep_environment(self, cache=True):
steps = []
if self.uses_apt():
if self.container:
steps += [
RunStep(
"set APT to non-interactive",
"echo 'debconf debconf/frontend select Noninteractive' | debconf-set-selections",
),
]
sudo = "sudo -n " if self.needs_sudo() else ""
steps += [
RunStep("Update APT", f"{sudo}apt update"),
]
if self.container:
if self.container == "centos:8":
steps += [
RunStep(
"Install config manager",
"dnf install -y 'dnf-command(config-manager)'",
),
RunStep(
"Enable PowerTools",
"dnf config-manager --set-enabled PowerTools",
),
]
steps += self.install_git()
steps += self.install_curl()
steps += [
CheckoutStep(),
# We need tags in order to use git describe for build/packaging
RunStep(
"Fetch tags", "git fetch --depth=1 origin +refs/tags/*:refs/tags/*"
),
RunStep("Fetch tag/branch history", "git fetch --prune --unshallow"),
]
steps += self.install_rust(cache="mac" not in self.name)
steps += self.install_system_deps()
return steps
def pull_request(self):
steps = self.prep_environment()
steps += self.check_formatting()
steps += self.build_all_release()
steps += self.test_all_release()
steps += self.package()
steps += self.upload_artifact()
return Job(
runs_on=self.os,
container=self.container,
steps=steps,
env=self.global_env(),
)
def continuous(self):
steps = self.prep_environment()
steps += self.build_all_release()
steps += self.test_all_release()
steps += self.package()
steps += self.upload_asset_nightly()
env = self.global_env()
env["BUILD_REASON"] = "Schedule"
return Job(
runs_on=self.os,
container=self.container,
steps=steps,
env=env,
)
def tag(self):
steps = self.prep_environment()
steps += self.build_all_release()
steps += self.test_all_release()
steps += self.package()
steps += self.upload_asset_tag()
steps += self.update_tagged_aur()
steps += self.update_homebrew_tap()
env = self.global_env()
return Job(
runs_on=self.os,
container=self.container,
steps=steps,
env=env,
)
TARGETS = [
Target(name="ubuntu:16", os="ubuntu-16.04", app_image=True),
Target(name="ubuntu:18", os="ubuntu-18.04", continuous_only=True),
Target(container="ubuntu:19.10", continuous_only=True),
# The container gets stuck while running get-deps, so disable for now
Target(container="ubuntu:20.04", continuous_only=True),
# debian 8's wayland libraries are too old for wayland-client
# Target(container="debian:8.11", continuous_only=True, bootstrap_git=True),
Target(container="debian:9.12", continuous_only=True, bootstrap_git=True),
Target(container="debian:10.3", continuous_only=True),
Target(name="macos", os="macos-latest"),
Target(container="fedora:31"),
Target(container="fedora:32"),
Target(container="fedora:33"),
Target(container="centos:7", bootstrap_git=True),
Target(container="centos:8"),
Target(name="windows", os="vs2017-win2016", rust_target="x86_64-pc-windows-msvc"),
]
def generate_actions(namer, jobber, trigger, is_continuous):
for t in TARGETS:
# if t.continuous_only and not is_continuous:
# continue
name = namer(t).replace(":", "")
print(name)
job = jobber(t)
file_name = f".github/workflows/gen_{name}.yml"
if job.container:
container = f"container: {yv(job.container)}"
else:
container = ""
with open(file_name, "w") as f:
f.write(
f"""
name: {name}
{trigger}
jobs:
build:
strategy:
fail-fast: false
runs-on: {yv(job.runs_on)}
{container}
steps:
"""
)
job.render(f)
# Sanity check the yaml, if pyyaml is available
try:
import yaml
with open(file_name) as f:
yaml.safe_load(f)
except ImportError:
pass
def generate_pr_actions():
generate_actions(
lambda t: f"{t.name}",
lambda t: t.pull_request(),
trigger="""
on:
push:
branches:
- master
pull_request:
branches:
- master
""",
is_continuous=False,
)
def continuous_actions():
generate_actions(
lambda t: f"{t.name}_continuous",
lambda t: t.continuous(),
trigger="""
on:
schedule:
- cron: "10 * * * *"
""",
is_continuous=True,
)
def tag_actions():
generate_actions(
lambda t: f"{t.name}_tag",
lambda t: t.tag(),
trigger="""
on:
push:
tags:
- "20*"
""",
is_continuous=True,
)
generate_pr_actions()
continuous_actions()
tag_actions()