forked from lix-project/hydra
Merge pull request #1025 from lukegb/hydra-better-errors
Produce better errors for failing jobsets
This commit is contained in:
commit
ef9a9fa481
4 changed files with 98 additions and 18 deletions
|
@ -1,6 +1,7 @@
|
||||||
#include <map>
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
|
#include <optional>
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
#include "shared.hh"
|
#include "shared.hh"
|
||||||
#include "store-api.hh"
|
#include "store-api.hh"
|
||||||
|
@ -442,11 +443,30 @@ int main(int argc, char * * argv)
|
||||||
auto named = job.find("namedConstituents");
|
auto named = job.find("namedConstituents");
|
||||||
if (named == job.end()) continue;
|
if (named == job.end()) continue;
|
||||||
|
|
||||||
|
std::unordered_map<std::string, std::string> brokenJobs;
|
||||||
|
auto getNonBrokenJobOrRecordError = [&brokenJobs, &jobName, &state](
|
||||||
|
const std::string & childJobName) -> std::optional<nlohmann::json> {
|
||||||
|
auto childJob = state->jobs.find(childJobName);
|
||||||
|
if (childJob == state->jobs.end()) {
|
||||||
|
printError("aggregate job '%s' references non-existent job '%s'", jobName, childJobName);
|
||||||
|
brokenJobs[childJobName] = "does not exist";
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
if (childJob->find("error") != childJob->end()) {
|
||||||
|
std::string error = (*childJob)["error"];
|
||||||
|
printError("aggregate job '%s' references broken job '%s': %s", jobName, childJobName, error);
|
||||||
|
brokenJobs[childJobName] = error;
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
return *childJob;
|
||||||
|
};
|
||||||
|
|
||||||
if (myArgs.dryRun) {
|
if (myArgs.dryRun) {
|
||||||
for (std::string jobName2 : *named) {
|
for (std::string jobName2 : *named) {
|
||||||
auto job2 = state->jobs.find(jobName2);
|
auto job2 = getNonBrokenJobOrRecordError(jobName2);
|
||||||
if (job2 == state->jobs.end())
|
if (!job2) {
|
||||||
throw Error("aggregate job '%s' references non-existent job '%s'", jobName, jobName2);
|
continue;
|
||||||
|
}
|
||||||
std::string drvPath2 = (*job2)["drvPath"];
|
std::string drvPath2 = (*job2)["drvPath"];
|
||||||
job["constituents"].push_back(drvPath2);
|
job["constituents"].push_back(drvPath2);
|
||||||
}
|
}
|
||||||
|
@ -455,15 +475,17 @@ int main(int argc, char * * argv)
|
||||||
auto drv = store->readDerivation(drvPath);
|
auto drv = store->readDerivation(drvPath);
|
||||||
|
|
||||||
for (std::string jobName2 : *named) {
|
for (std::string jobName2 : *named) {
|
||||||
auto job2 = state->jobs.find(jobName2);
|
auto job2 = getNonBrokenJobOrRecordError(jobName2);
|
||||||
if (job2 == state->jobs.end())
|
if (!job2) {
|
||||||
throw Error("aggregate job '%s' references non-existent job '%s'", jobName, jobName2);
|
continue;
|
||||||
|
}
|
||||||
auto drvPath2 = store->parseStorePath((std::string) (*job2)["drvPath"]);
|
auto drvPath2 = store->parseStorePath((std::string) (*job2)["drvPath"]);
|
||||||
auto drv2 = store->readDerivation(drvPath2);
|
auto drv2 = store->readDerivation(drvPath2);
|
||||||
job["constituents"].push_back(store->printStorePath(drvPath2));
|
job["constituents"].push_back(store->printStorePath(drvPath2));
|
||||||
drv.inputDrvs[drvPath2] = {drv2.outputs.begin()->first};
|
drv.inputDrvs[drvPath2] = {drv2.outputs.begin()->first};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (brokenJobs.empty()) {
|
||||||
std::string drvName(drvPath.name());
|
std::string drvName(drvPath.name());
|
||||||
assert(hasSuffix(drvName, drvExtension));
|
assert(hasSuffix(drvName, drvExtension));
|
||||||
drvName.resize(drvName.size() - drvExtension.size());
|
drvName.resize(drvName.size() - drvExtension.size());
|
||||||
|
@ -478,8 +500,17 @@ int main(int argc, char * * argv)
|
||||||
job["drvPath"] = newDrvPath;
|
job["drvPath"] = newDrvPath;
|
||||||
job["outputs"]["out"] = store->printStorePath(outPath);
|
job["outputs"]["out"] = store->printStorePath(outPath);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
job.erase("namedConstituents");
|
job.erase("namedConstituents");
|
||||||
|
|
||||||
|
if (!brokenJobs.empty()) {
|
||||||
|
std::stringstream ss;
|
||||||
|
for (const auto& [jobName, error] : brokenJobs) {
|
||||||
|
ss << jobName << ": " << error << "\n";
|
||||||
|
}
|
||||||
|
job["error"] = ss.str();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::cout << state->jobs.dump(2) << "\n";
|
std::cout << state->jobs.dump(2) << "\n";
|
||||||
|
|
16
t/jobs/broken-constituent.nix
Normal file
16
t/jobs/broken-constituent.nix
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
with import ./config.nix;
|
||||||
|
{
|
||||||
|
broken = mkDerivation {
|
||||||
|
name = "broken";
|
||||||
|
_hydraAggregate = true;
|
||||||
|
constituents = [
|
||||||
|
"does-not-exist"
|
||||||
|
"does-not-evaluate"
|
||||||
|
];
|
||||||
|
builder = ./fail.sh;
|
||||||
|
};
|
||||||
|
|
||||||
|
# does-not-exist doesn't exist.
|
||||||
|
|
||||||
|
does-not-evaluate = assert false; {};
|
||||||
|
}
|
|
@ -148,6 +148,7 @@ sub createJobsetWithOneInput {
|
||||||
sub evalSucceeds {
|
sub evalSucceeds {
|
||||||
my ($jobset) = @_;
|
my ($jobset) = @_;
|
||||||
my ($res, $stdout, $stderr) = captureStdoutStderr(60, ("hydra-eval-jobset", $jobset->project->name, $jobset->name));
|
my ($res, $stdout, $stderr) = captureStdoutStderr(60, ("hydra-eval-jobset", $jobset->project->name, $jobset->name));
|
||||||
|
$jobset->discard_changes; # refresh from DB
|
||||||
chomp $stdout; chomp $stderr;
|
chomp $stdout; chomp $stderr;
|
||||||
print STDERR "Evaluation errors for jobset ".$jobset->project->name.":".$jobset->name.": \n".$jobset->errormsg."\n" if $jobset->errormsg;
|
print STDERR "Evaluation errors for jobset ".$jobset->project->name.":".$jobset->name.": \n".$jobset->errormsg."\n" if $jobset->errormsg;
|
||||||
print STDERR "STDOUT: $stdout\n" if $stdout ne "";
|
print STDERR "STDOUT: $stdout\n" if $stdout ne "";
|
||||||
|
|
32
t/queue-runner/constituents.t
Normal file
32
t/queue-runner/constituents.t
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
use feature 'unicode_strings';
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Setup;
|
||||||
|
|
||||||
|
my %ctx = test_init();
|
||||||
|
|
||||||
|
require Hydra::Schema;
|
||||||
|
require Hydra::Model::DB;
|
||||||
|
|
||||||
|
use Test2::V0;
|
||||||
|
|
||||||
|
my $db = Hydra::Model::DB->new;
|
||||||
|
hydra_setup($db);
|
||||||
|
|
||||||
|
my $project = $db->resultset('Projects')->create({name => "tests", displayname => "", owner => "root"});
|
||||||
|
|
||||||
|
my $jobset = createBaseJobset("broken-constituent", "broken-constituent.nix", $ctx{jobsdir});
|
||||||
|
|
||||||
|
ok(evalSucceeds($jobset), "Evaluating jobs/broken-constituent.nix should exit with return code 0");
|
||||||
|
is(nrQueuedBuildsForJobset($jobset), 0, "Evaluating jobs/broken-constituent.nix should not queue any builds");
|
||||||
|
|
||||||
|
like(
|
||||||
|
$jobset->errormsg,
|
||||||
|
qr/^does-not-exist: does not exist$/m,
|
||||||
|
"Evaluating jobs/broken-constituent.nix should log an error for does-not-exist");
|
||||||
|
like(
|
||||||
|
$jobset->errormsg,
|
||||||
|
qr/^does-not-evaluate: error: assertion 'false' failed$/m,
|
||||||
|
"Evaluating jobs/broken-constituent.nix should log an error for does-not-evaluate");
|
||||||
|
|
||||||
|
done_testing;
|
Loading…
Reference in a new issue