Merge pull request #39 from Mic92/fixes
Move skipped builds to local worker
This commit is contained in:
commit
d614a8806c
5 changed files with 106 additions and 67 deletions
|
@ -31,6 +31,8 @@ from .github_projects import ( # noqa: E402
|
||||||
slugify_project_name,
|
slugify_project_name,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
SKIPPED_BUILDER_NAME = "skipped-builds"
|
||||||
|
|
||||||
|
|
||||||
class BuildTrigger(Trigger):
|
class BuildTrigger(Trigger):
|
||||||
"""
|
"""
|
||||||
|
@ -38,16 +40,22 @@ class BuildTrigger(Trigger):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self, scheduler: str, jobs: list[dict[str, Any]], **kwargs: Any
|
self,
|
||||||
|
builds_scheduler: str,
|
||||||
|
skipped_builds_scheduler: str,
|
||||||
|
jobs: list[dict[str, Any]],
|
||||||
|
**kwargs: Any,
|
||||||
) -> None:
|
) -> None:
|
||||||
if "name" not in kwargs:
|
if "name" not in kwargs:
|
||||||
kwargs["name"] = "trigger"
|
kwargs["name"] = "trigger"
|
||||||
self.jobs = jobs
|
self.jobs = jobs
|
||||||
self.config = None
|
self.config = None
|
||||||
|
self.builds_scheduler = builds_scheduler
|
||||||
|
self.skipped_builds_scheduler = skipped_builds_scheduler
|
||||||
Trigger.__init__(
|
Trigger.__init__(
|
||||||
self,
|
self,
|
||||||
waitForFinish=True,
|
waitForFinish=True,
|
||||||
schedulerNames=[scheduler],
|
schedulerNames=[builds_scheduler, skipped_builds_scheduler],
|
||||||
haltOnFailure=True,
|
haltOnFailure=True,
|
||||||
flunkOnFailure=True,
|
flunkOnFailure=True,
|
||||||
sourceStamps=[],
|
sourceStamps=[],
|
||||||
|
@ -68,7 +76,6 @@ class BuildTrigger(Trigger):
|
||||||
project_id = slugify_project_name(repo_name)
|
project_id = slugify_project_name(repo_name)
|
||||||
source = f"nix-eval-{project_id}"
|
source = f"nix-eval-{project_id}"
|
||||||
|
|
||||||
sch = self.schedulerNames[0]
|
|
||||||
triggered_schedulers = []
|
triggered_schedulers = []
|
||||||
for job in self.jobs:
|
for job in self.jobs:
|
||||||
attr = job.get("attr", "eval-error")
|
attr = job.get("attr", "eval-error")
|
||||||
|
@ -77,28 +84,38 @@ class BuildTrigger(Trigger):
|
||||||
name = f"github:{repo_name}#checks.{name}"
|
name = f"github:{repo_name}#checks.{name}"
|
||||||
else:
|
else:
|
||||||
name = f"checks.{name}"
|
name = f"checks.{name}"
|
||||||
drv_path = job.get("drvPath")
|
|
||||||
error = job.get("error")
|
error = job.get("error")
|
||||||
|
props = Properties()
|
||||||
|
props.setProperty("virtual_builder_tags", "", source)
|
||||||
|
|
||||||
|
if error is not None:
|
||||||
|
props.setProperty("error", error, source)
|
||||||
|
props.setProperty("virtual_builder_name", f"{name}", source)
|
||||||
|
triggered_schedulers.append((self.skipped_builds_scheduler, props))
|
||||||
|
continue
|
||||||
|
|
||||||
|
if job.get("isCached"):
|
||||||
|
props.setProperty("virtual_builder_name", f"{name}", source)
|
||||||
|
triggered_schedulers.append((self.skipped_builds_scheduler, props))
|
||||||
|
continue
|
||||||
|
|
||||||
|
drv_path = job.get("drvPath")
|
||||||
system = job.get("system")
|
system = job.get("system")
|
||||||
out_path = job.get("outputs", {}).get("out")
|
out_path = job.get("outputs", {}).get("out")
|
||||||
|
|
||||||
build_props.setProperty(f"{attr}-out_path", out_path, source)
|
build_props.setProperty(f"{attr}-out_path", out_path, source)
|
||||||
build_props.setProperty(f"{attr}-drv_path", drv_path, source)
|
build_props.setProperty(f"{attr}-drv_path", drv_path, source)
|
||||||
|
|
||||||
props = Properties()
|
|
||||||
props.setProperty("virtual_builder_name", name, source)
|
props.setProperty("virtual_builder_name", name, source)
|
||||||
props.setProperty("status_name", f"nix-build .#checks.{attr}", source)
|
props.setProperty("status_name", f"nix-build .#checks.{attr}", source)
|
||||||
props.setProperty("virtual_builder_tags", "", source)
|
|
||||||
props.setProperty("attr", attr, source)
|
props.setProperty("attr", attr, source)
|
||||||
props.setProperty("system", system, source)
|
props.setProperty("system", system, source)
|
||||||
props.setProperty("drv_path", drv_path, source)
|
props.setProperty("drv_path", drv_path, source)
|
||||||
props.setProperty("out_path", out_path, source)
|
props.setProperty("out_path", out_path, source)
|
||||||
# we use this to identify builds when running a retry
|
# we use this to identify builds when running a retry
|
||||||
props.setProperty("build_uuid", str(uuid.uuid4()), source)
|
props.setProperty("build_uuid", str(uuid.uuid4()), source)
|
||||||
props.setProperty("error", error, source)
|
|
||||||
props.setProperty("is_cached", job.get("isCached"), source)
|
|
||||||
|
|
||||||
triggered_schedulers.append((sch, props))
|
triggered_schedulers.append((self.builds_scheduler, props))
|
||||||
return triggered_schedulers
|
return triggered_schedulers
|
||||||
|
|
||||||
def getCurrentSummary(self) -> dict[str, str]: # noqa: N802
|
def getCurrentSummary(self) -> dict[str, str]: # noqa: N802
|
||||||
|
@ -156,7 +173,6 @@ class NixEvalCommand(buildstep.ShellMixin, steps.BuildStep):
|
||||||
build_props.getProperty("github.repository.full_name"),
|
build_props.getProperty("github.repository.full_name"),
|
||||||
)
|
)
|
||||||
project_id = slugify_project_name(repo_name)
|
project_id = slugify_project_name(repo_name)
|
||||||
scheduler = f"{project_id}-nix-build"
|
|
||||||
filtered_jobs = []
|
filtered_jobs = []
|
||||||
for job in jobs:
|
for job in jobs:
|
||||||
system = job.get("system")
|
system = job.get("system")
|
||||||
|
@ -168,7 +184,10 @@ class NixEvalCommand(buildstep.ShellMixin, steps.BuildStep):
|
||||||
self.build.addStepsAfterCurrentStep(
|
self.build.addStepsAfterCurrentStep(
|
||||||
[
|
[
|
||||||
BuildTrigger(
|
BuildTrigger(
|
||||||
scheduler=scheduler, name="build flake", jobs=filtered_jobs
|
builds_scheduler=f"{project_id}-nix-build",
|
||||||
|
skipped_builds_scheduler=f"{project_id}-nix-skipped-build",
|
||||||
|
name="build flake",
|
||||||
|
jobs=filtered_jobs,
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
@ -195,35 +214,32 @@ class RetryCounter:
|
||||||
RETRY_COUNTER = RetryCounter(retries=2)
|
RETRY_COUNTER = RetryCounter(retries=2)
|
||||||
|
|
||||||
|
|
||||||
|
class EvalErrorStep(steps.BuildStep):
|
||||||
|
"""
|
||||||
|
Shows the error message of a failed evaluation.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@defer.inlineCallbacks
|
||||||
|
def run(self) -> Generator[Any, object, Any]:
|
||||||
|
error = self.getProperty("error")
|
||||||
|
attr = self.getProperty("attr")
|
||||||
|
# show eval error
|
||||||
|
error_log: Log = yield self.addLog("nix_error")
|
||||||
|
error_log.addStderr(f"{attr} failed to evaluate:\n{error}")
|
||||||
|
return util.FAILURE
|
||||||
|
|
||||||
|
|
||||||
class NixBuildCommand(buildstep.ShellMixin, steps.BuildStep):
|
class NixBuildCommand(buildstep.ShellMixin, steps.BuildStep):
|
||||||
"""
|
"""
|
||||||
Builds a nix derivation if evaluation was successful,
|
Builds a nix derivation.
|
||||||
otherwise this shows the evaluation error.
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, **kwargs: Any) -> None:
|
def __init__(self, **kwargs: Any) -> None:
|
||||||
kwargs = self.setupShellMixin(kwargs)
|
kwargs = self.setupShellMixin(kwargs)
|
||||||
super().__init__(**kwargs)
|
super().__init__(**kwargs)
|
||||||
self.observer = logobserver.BufferLogObserver()
|
|
||||||
self.addLogObserver("stdio", self.observer)
|
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
@defer.inlineCallbacks
|
||||||
def run(self) -> Generator[Any, object, Any]:
|
def run(self) -> Generator[Any, object, Any]:
|
||||||
error = self.getProperty("error")
|
|
||||||
if error is not None:
|
|
||||||
attr = self.getProperty("attr")
|
|
||||||
# show eval error
|
|
||||||
self.build.results = util.FAILURE
|
|
||||||
error_log: Log = yield self.addLog("nix_error")
|
|
||||||
error_log.addStderr(f"{attr} failed to evaluate:\n{error}")
|
|
||||||
return util.FAILURE
|
|
||||||
|
|
||||||
cached = self.getProperty("is_cached")
|
|
||||||
if cached:
|
|
||||||
log: Log = yield self.addLog("log")
|
|
||||||
log.addStderr("Build is already the binary cache.")
|
|
||||||
return util.SKIPPED
|
|
||||||
|
|
||||||
# run `nix build`
|
# run `nix build`
|
||||||
cmd: remotecommand.RemoteCommand = yield self.makeRemoteShellCommand()
|
cmd: remotecommand.RemoteCommand = yield self.makeRemoteShellCommand()
|
||||||
yield self.runCommand(cmd)
|
yield self.runCommand(cmd)
|
||||||
|
@ -254,10 +270,6 @@ class UpdateBuildOutput(steps.BuildStep):
|
||||||
):
|
):
|
||||||
return util.SKIPPED
|
return util.SKIPPED
|
||||||
|
|
||||||
cached = props.getProperty("is_cached")
|
|
||||||
if cached:
|
|
||||||
return util.SKIPPED
|
|
||||||
|
|
||||||
attr = os.path.basename(props.getProperty("attr"))
|
attr = os.path.basename(props.getProperty("attr"))
|
||||||
out_path = props.getProperty("out_path")
|
out_path = props.getProperty("out_path")
|
||||||
# XXX don't hardcode this
|
# XXX don't hardcode this
|
||||||
|
@ -528,7 +540,6 @@ def nix_build_config(
|
||||||
util.Secret("cachix-name"),
|
util.Secret("cachix-name"),
|
||||||
util.Interpolate("result-%(prop:attr)s"),
|
util.Interpolate("result-%(prop:attr)s"),
|
||||||
],
|
],
|
||||||
doStepIf=lambda s: not s.getProperty("is_cached"),
|
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -545,8 +556,7 @@ def nix_build_config(
|
||||||
"-r",
|
"-r",
|
||||||
util.Property("out_path"),
|
util.Property("out_path"),
|
||||||
],
|
],
|
||||||
doStepIf=lambda s: not s.getProperty("is_cached")
|
doStepIf=lambda s: s.getProperty("branch")
|
||||||
and s.getProperty("branch")
|
|
||||||
== s.getProperty("github.repository.default_branch"),
|
== s.getProperty("github.repository.default_branch"),
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
@ -554,7 +564,6 @@ def nix_build_config(
|
||||||
steps.ShellCommand(
|
steps.ShellCommand(
|
||||||
name="Delete temporary gcroots",
|
name="Delete temporary gcroots",
|
||||||
command=["rm", "-f", util.Interpolate("result-%(prop:attr)s")],
|
command=["rm", "-f", util.Interpolate("result-%(prop:attr)s")],
|
||||||
doStepIf=lambda s: not s.getProperty("is_cached"),
|
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
if outputs_path is not None:
|
if outputs_path is not None:
|
||||||
|
@ -574,6 +583,39 @@ def nix_build_config(
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def nix_skipped_build_config(
|
||||||
|
project: GithubProject, worker_names: list[str]
|
||||||
|
) -> util.BuilderConfig:
|
||||||
|
"""
|
||||||
|
Dummy builder that is triggered when a build is skipped.
|
||||||
|
"""
|
||||||
|
factory = util.BuildFactory()
|
||||||
|
factory.addStep(
|
||||||
|
EvalErrorStep(
|
||||||
|
name="Nix evaluation",
|
||||||
|
doStepIf=lambda s: s.getProperty("error"),
|
||||||
|
hideStepIf=lambda _, s: not s.getProperty("error"),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
# This is just a dummy step showing the cached build
|
||||||
|
factory.addStep(
|
||||||
|
steps.BuildStep(
|
||||||
|
name="Nix build (cached)",
|
||||||
|
doStepIf=lambda _: False,
|
||||||
|
hideStepIf=lambda _, s: s.getProperty("error"),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return util.BuilderConfig(
|
||||||
|
name=f"{project.name}/nix-skipped-build",
|
||||||
|
project=project.name,
|
||||||
|
workernames=worker_names,
|
||||||
|
collapseRequests=False,
|
||||||
|
env={},
|
||||||
|
factory=factory,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def read_secret_file(secret_name: str) -> str:
|
def read_secret_file(secret_name: str) -> str:
|
||||||
directory = os.environ.get("CREDENTIALS_DIRECTORY")
|
directory = os.environ.get("CREDENTIALS_DIRECTORY")
|
||||||
if directory is None:
|
if directory is None:
|
||||||
|
@ -609,11 +651,6 @@ def config_for_project(
|
||||||
eval_lock: util.WorkerLock,
|
eval_lock: util.WorkerLock,
|
||||||
outputs_path: Path | None = None,
|
outputs_path: Path | None = None,
|
||||||
) -> Project:
|
) -> Project:
|
||||||
## get a deterministic jitter for the project
|
|
||||||
# random.seed(project.name)
|
|
||||||
## don't run all projects at the same time
|
|
||||||
# jitter = random.randint(1, 60) * 60
|
|
||||||
|
|
||||||
config["projects"].append(Project(project.name))
|
config["projects"].append(Project(project.name))
|
||||||
config["schedulers"].extend(
|
config["schedulers"].extend(
|
||||||
[
|
[
|
||||||
|
@ -649,6 +686,11 @@ def config_for_project(
|
||||||
name=f"{project.id}-nix-build",
|
name=f"{project.id}-nix-build",
|
||||||
builderNames=[f"{project.name}/nix-build"],
|
builderNames=[f"{project.name}/nix-build"],
|
||||||
),
|
),
|
||||||
|
# this is triggered from `nix-eval` when the build is skipped
|
||||||
|
schedulers.Triggerable(
|
||||||
|
name=f"{project.id}-nix-skipped-build",
|
||||||
|
builderNames=[f"{project.name}/nix-skipped-build"],
|
||||||
|
),
|
||||||
# allow to manually trigger a nix-build
|
# allow to manually trigger a nix-build
|
||||||
schedulers.ForceScheduler(
|
schedulers.ForceScheduler(
|
||||||
name=f"{project.id}-force",
|
name=f"{project.id}-force",
|
||||||
|
@ -674,12 +716,6 @@ def config_for_project(
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
# updates flakes once a week
|
|
||||||
# schedulers.Periodic(
|
|
||||||
# name=f"{project.id}-update-flake-weekly",
|
|
||||||
# builderNames=[f"{project.name}/update-flake"],
|
|
||||||
# periodicBuildTimer=24 * 60 * 60 * 7 + jitter,
|
|
||||||
# ),
|
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
has_cachix_auth_token = os.path.isfile(
|
has_cachix_auth_token = os.path.isfile(
|
||||||
|
@ -708,6 +744,7 @@ def config_for_project(
|
||||||
has_cachix_signing_key,
|
has_cachix_signing_key,
|
||||||
outputs_path=outputs_path,
|
outputs_path=outputs_path,
|
||||||
),
|
),
|
||||||
|
nix_skipped_build_config(project, [SKIPPED_BUILDER_NAME]),
|
||||||
nix_update_flake_config(
|
nix_update_flake_config(
|
||||||
project,
|
project,
|
||||||
worker_names,
|
worker_names,
|
||||||
|
@ -798,6 +835,7 @@ class NixConfigurator(ConfiguratorBase):
|
||||||
self.github.project_cache_file,
|
self.github.project_cache_file,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
config["workers"].append(worker.LocalWorker(SKIPPED_BUILDER_NAME))
|
||||||
config["schedulers"].extend(
|
config["schedulers"].extend(
|
||||||
[
|
[
|
||||||
schedulers.ForceScheduler(
|
schedulers.ForceScheduler(
|
||||||
|
|
20
flake.lock
20
flake.lock
|
@ -7,11 +7,11 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1698579227,
|
"lastModified": 1698882062,
|
||||||
"narHash": "sha256-KVWjFZky+gRuWennKsbo6cWyo7c/z/VgCte5pR9pEKg=",
|
"narHash": "sha256-HkhafUayIqxXyHH1X8d9RDl1M2CkFgZLjKD3MzabiEo=",
|
||||||
"owner": "hercules-ci",
|
"owner": "hercules-ci",
|
||||||
"repo": "flake-parts",
|
"repo": "flake-parts",
|
||||||
"rev": "f76e870d64779109e41370848074ac4eaa1606ec",
|
"rev": "8c9fa2545007b49a5db5f650ae91f227672c3877",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
@ -22,16 +22,16 @@
|
||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1698653593,
|
"lastModified": 1700292038,
|
||||||
"narHash": "sha256-4SW5hJ7ktIO6j1+aNah0c9u+XDxjR4uYwPVtkVZynrs=",
|
"narHash": "sha256-dD/6dw/i0GjweIf3v6F9lZdsU9VaLMiQEX9XL8ruryk=",
|
||||||
"owner": "Nixos",
|
"owner": "Nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "423b31f1b24ec8d82baec9a5bb969da892010e6d",
|
"rev": "ef355dde3c80e7ee30aa65aa5bf76e0b1b00bcc2",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
"owner": "Nixos",
|
"owner": "Nixos",
|
||||||
"ref": "nixos-unstable-small",
|
"ref": "master",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
|
@ -50,11 +50,11 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1698438538,
|
"lastModified": 1699786194,
|
||||||
"narHash": "sha256-AWxaKTDL3MtxaVTVU5lYBvSnlspOS0Fjt8GxBgnU0Do=",
|
"narHash": "sha256-3h3EH1FXQkIeAuzaWB+nK0XK54uSD46pp+dMD3gAcB4=",
|
||||||
"owner": "numtide",
|
"owner": "numtide",
|
||||||
"repo": "treefmt-nix",
|
"repo": "treefmt-nix",
|
||||||
"rev": "5deb8dc125a9f83b65ca86cf0c8167c46593e0b1",
|
"rev": "e82f32aa7f06bbbd56d7b12186d555223dc399d1",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
|
|
@ -3,7 +3,10 @@
|
||||||
description = "A nixos module to make buildbot a proper Nix-CI.";
|
description = "A nixos module to make buildbot a proper Nix-CI.";
|
||||||
|
|
||||||
inputs = {
|
inputs = {
|
||||||
nixpkgs.url = "github:Nixos/nixpkgs/nixos-unstable-small";
|
nixpkgs.url = "github:Nixos/nixpkgs/master";
|
||||||
|
# switch back later when https://github.com/NixOS/nixpkgs/pull/266270 is included:
|
||||||
|
# see: https://nixpk.gs/pr-tracker.html?pr=266270
|
||||||
|
#nixpkgs.url = "github:Nixos/nixpkgs/nixos-unstable-small";
|
||||||
flake-parts.url = "github:hercules-ci/flake-parts";
|
flake-parts.url = "github:hercules-ci/flake-parts";
|
||||||
flake-parts.inputs.nixpkgs-lib.follows = "nixpkgs";
|
flake-parts.inputs.nixpkgs-lib.follows = "nixpkgs";
|
||||||
|
|
||||||
|
|
|
@ -209,12 +209,10 @@ in
|
||||||
services.postgresql = {
|
services.postgresql = {
|
||||||
enable = true;
|
enable = true;
|
||||||
ensureDatabases = [ "buildbot" ];
|
ensureDatabases = [ "buildbot" ];
|
||||||
ensureUsers = [
|
ensureUsers = [{
|
||||||
{
|
name = "buildbot";
|
||||||
name = "buildbot";
|
ensureDBOwnership = true;
|
||||||
ensurePermissions."DATABASE buildbot" = "ALL PRIVILEGES";
|
}];
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
services.nginx.enable = true;
|
services.nginx.enable = true;
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
"-eucx"
|
"-eucx"
|
||||||
''
|
''
|
||||||
${pkgs.ruff}/bin/ruff --fix "$@"
|
${pkgs.ruff}/bin/ruff --fix "$@"
|
||||||
${pkgs.python3.pkgs.black}/bin/black "$@"
|
${pkgs.ruff}/bin/ruff format "$@"
|
||||||
''
|
''
|
||||||
"--" # this argument is ignored by bash
|
"--" # this argument is ignored by bash
|
||||||
];
|
];
|
||||||
|
|
Loading…
Reference in a new issue