2008-11-25 11:01:42 +00:00
|
|
|
|
package Hydra::Helper::Nix;
|
2008-11-18 14:48:40 +00:00
|
|
|
|
|
|
|
|
|
use strict;
|
2008-11-28 14:36:04 +00:00
|
|
|
|
use Exporter;
|
2009-02-06 21:01:20 +00:00
|
|
|
|
use File::Path;
|
2009-01-13 14:02:07 +00:00
|
|
|
|
use File::Basename;
|
2013-05-09 14:29:04 +02:00
|
|
|
|
use Config::General;
|
2010-09-03 09:17:52 +00:00
|
|
|
|
use Hydra::Helper::CatalystUtils;
|
2012-03-13 12:10:19 +01:00
|
|
|
|
use Hydra::Model::DB;
|
2013-10-04 17:01:47 +02:00
|
|
|
|
use Nix::Store;
|
2014-08-13 18:53:29 +02:00
|
|
|
|
use Encode;
|
2008-11-28 14:36:04 +00:00
|
|
|
|
|
|
|
|
|
our @ISA = qw(Exporter);
|
2009-02-06 15:02:49 +00:00
|
|
|
|
our @EXPORT = qw(
|
2013-01-22 13:19:28 +01:00
|
|
|
|
getHydraHome getHydraConfig txn_do
|
2013-05-25 15:36:58 -04:00
|
|
|
|
getSCMCacheDir
|
2009-03-15 11:56:11 +00:00
|
|
|
|
registerRoot getGCRootsDir gcRootFor
|
2013-11-05 16:05:29 +01:00
|
|
|
|
jobsetOverview jobsetOverview_
|
|
|
|
|
removeAsciiEscapes getDrvLogPath findLog logContents
|
2013-02-21 17:27:17 +01:00
|
|
|
|
getMainOutput
|
2013-04-02 23:32:04 +02:00
|
|
|
|
getEvals getMachines
|
2013-05-25 15:36:58 -04:00
|
|
|
|
pathIsInsidePrefix
|
2013-09-21 14:47:52 +00:00
|
|
|
|
captureStdoutStderr run grab
|
2013-10-04 15:40:43 +02:00
|
|
|
|
getTotalShares
|
2013-10-04 17:01:47 +02:00
|
|
|
|
cancelBuilds restartBuilds);
|
2008-11-18 14:48:40 +00:00
|
|
|
|
|
|
|
|
|
|
2012-02-28 20:16:16 +01:00
|
|
|
|
sub getHydraHome {
|
|
|
|
|
my $dir = $ENV{"HYDRA_HOME"} or die "The HYDRA_HOME directory does not exist!\n";
|
|
|
|
|
return $dir;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-01-22 13:19:28 +01:00
|
|
|
|
sub getHydraConfig {
|
2012-03-13 12:10:19 +01:00
|
|
|
|
my $conf = $ENV{"HYDRA_CONFIG"} || (Hydra::Model::DB::getHydraPath . "/hydra.conf");
|
2013-01-22 13:19:28 +01:00
|
|
|
|
return {} unless -f $conf;
|
|
|
|
|
my %config = new Config::General($conf)->getall;
|
|
|
|
|
return \%config;
|
2011-03-07 15:06:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-28 14:36:04 +00:00
|
|
|
|
|
2009-04-22 22:43:04 +00:00
|
|
|
|
# Awful hack to handle timeouts in SQLite: just retry the transaction.
|
|
|
|
|
# DBD::SQLite *has* a 30 second retry window, but apparently it
|
|
|
|
|
# doesn't work.
|
|
|
|
|
sub txn_do {
|
|
|
|
|
my ($db, $coderef) = @_;
|
2013-10-04 15:40:43 +02:00
|
|
|
|
my $res;
|
2009-04-22 22:43:04 +00:00
|
|
|
|
while (1) {
|
|
|
|
|
eval {
|
2013-10-04 15:40:43 +02:00
|
|
|
|
$res = $db->txn_do($coderef);
|
2009-04-22 22:43:04 +00:00
|
|
|
|
};
|
2013-10-04 15:40:43 +02:00
|
|
|
|
return $res if !$@;
|
2009-04-22 22:43:04 +00:00
|
|
|
|
die $@ unless $@ =~ "database is locked";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-05-25 15:36:58 -04:00
|
|
|
|
sub getSCMCacheDir {
|
|
|
|
|
return Hydra::Model::DB::getHydraPath . "/scm" ;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-02-06 21:01:20 +00:00
|
|
|
|
sub getGCRootsDir {
|
|
|
|
|
die unless defined $ENV{LOGNAME};
|
2011-04-18 08:10:10 +00:00
|
|
|
|
my $dir = ($ENV{NIX_STATE_DIR} || "/nix/var/nix" ) . "/gcroots/per-user/$ENV{LOGNAME}/hydra-roots";
|
2009-03-15 11:56:11 +00:00
|
|
|
|
mkpath $dir if !-e $dir;
|
|
|
|
|
return $dir;
|
2009-02-06 21:01:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-15 11:56:11 +00:00
|
|
|
|
sub gcRootFor {
|
2009-02-06 21:01:20 +00:00
|
|
|
|
my ($path) = @_;
|
2009-03-31 15:59:47 +00:00
|
|
|
|
return getGCRootsDir . "/" . basename $path;
|
2009-03-15 11:56:11 +00:00
|
|
|
|
}
|
2009-02-06 21:01:20 +00:00
|
|
|
|
|
|
|
|
|
|
2009-03-15 11:56:11 +00:00
|
|
|
|
sub registerRoot {
|
|
|
|
|
my ($path) = @_;
|
|
|
|
|
my $link = gcRootFor $path;
|
2014-08-13 16:29:00 +02:00
|
|
|
|
return if -e $link;
|
2014-08-01 17:24:55 +02:00
|
|
|
|
open ROOT, ">$link" or die "cannot create GC root `$link' to `$path'";
|
|
|
|
|
close ROOT;
|
2009-02-06 21:01:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
|
sub attrsToSQL {
|
|
|
|
|
my ($attrs, $id) = @_;
|
|
|
|
|
my @attrs = split / /, $attrs;
|
|
|
|
|
|
|
|
|
|
my $query = "1 = 1";
|
|
|
|
|
|
|
|
|
|
foreach my $attr (@attrs) {
|
2009-10-20 12:35:01 +00:00
|
|
|
|
$attr =~ /^([\w-]+)=([\w-]*)$/ or die "invalid attribute in view: $attr";
|
2009-02-06 15:02:49 +00:00
|
|
|
|
my $name = $1;
|
|
|
|
|
my $value = $2;
|
|
|
|
|
# !!! Yes, this is horribly injection-prone... (though
|
|
|
|
|
# name/value are filtered above). Should use SQL::Abstract,
|
|
|
|
|
# but it can't deal with subqueries. At least we should use
|
|
|
|
|
# placeholders.
|
2009-03-20 14:50:09 +00:00
|
|
|
|
$query .= " and exists (select 1 from buildinputs where build = $id and name = '$name' and value = '$value')";
|
2009-02-06 15:02:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $query;
|
|
|
|
|
}
|
|
|
|
|
|
2012-03-05 21:52:47 +01:00
|
|
|
|
|
2013-11-05 16:05:29 +01:00
|
|
|
|
sub jobsetOverview_ {
|
|
|
|
|
my ($c, $jobsets) = @_;
|
|
|
|
|
return $jobsets->search({},
|
2011-08-19 15:13:34 +00:00
|
|
|
|
{ order_by => "name"
|
2012-06-26 12:00:18 +02:00
|
|
|
|
, "+select" =>
|
2012-02-29 02:22:49 +01:00
|
|
|
|
[ "(select count(*) from Builds as a where a.finished = 0 and me.project = a.project and me.name = a.jobset and a.isCurrent = 1)"
|
2012-03-05 21:52:47 +01:00
|
|
|
|
, "(select count(*) from Builds as a where a.finished = 1 and me.project = a.project and me.name = a.jobset and buildstatus <> 0 and a.isCurrent = 1)"
|
|
|
|
|
, "(select count(*) from Builds as a where a.finished = 1 and me.project = a.project and me.name = a.jobset and buildstatus = 0 and a.isCurrent = 1)"
|
2012-02-29 02:22:49 +01:00
|
|
|
|
, "(select count(*) from Builds as a where me.project = a.project and me.name = a.jobset and a.isCurrent = 1)"
|
2011-08-19 15:13:34 +00:00
|
|
|
|
]
|
2013-01-22 14:09:37 +01:00
|
|
|
|
, "+as" => ["nrscheduled", "nrfailed", "nrsucceeded", "nrtotal"]
|
2012-02-29 02:22:49 +01:00
|
|
|
|
});
|
2010-09-03 09:17:52 +00:00
|
|
|
|
}
|
2009-10-20 12:35:01 +00:00
|
|
|
|
|
2012-03-05 21:52:47 +01:00
|
|
|
|
|
2013-11-05 16:05:29 +01:00
|
|
|
|
sub jobsetOverview {
|
|
|
|
|
my ($c, $project) = @_;
|
|
|
|
|
my $jobsets = $project->jobsets->search(isProjectOwner($c, $project) ? {} : { hidden => 0 });
|
|
|
|
|
return jobsetOverview_($c, $jobsets);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-01-22 22:48:02 +01:00
|
|
|
|
# Return the path of the build log of the given derivation, or undef
|
|
|
|
|
# if the log is gone.
|
|
|
|
|
sub getDrvLogPath {
|
|
|
|
|
my ($drvPath) = @_;
|
|
|
|
|
my $base = basename $drvPath;
|
2013-02-13 13:09:07 +01:00
|
|
|
|
my $bucketed = substr($base, 0, 2) . "/" . substr($base, 2);
|
|
|
|
|
my $fn = ($ENV{NIX_LOG_DIR} || "/nix/var/log/nix") . "/drvs/";
|
2015-06-09 14:21:21 +02:00
|
|
|
|
my $fn2 = Hydra::Model::DB::getHydraPath . "/build-logs/";
|
|
|
|
|
for ($fn2 . $bucketed, $fn . $bucketed . ".bz2", $fn . $bucketed, $fn . $base . ".bz2", $fn . $base) {
|
|
|
|
|
return $_ if -f $_;
|
2013-02-13 13:09:07 +01:00
|
|
|
|
}
|
2013-01-22 22:48:02 +01:00
|
|
|
|
return undef;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-08-30 13:53:25 +00:00
|
|
|
|
# Find the log of the derivation denoted by $drvPath. It it doesn't
|
|
|
|
|
# exist, try other derivations that produced its outputs (@outPaths).
|
|
|
|
|
sub findLog {
|
|
|
|
|
my ($c, $drvPath, @outPaths) = @_;
|
|
|
|
|
|
|
|
|
|
if (defined $drvPath) {
|
2013-08-30 18:11:03 +00:00
|
|
|
|
my $logPath = getDrvLogPath($drvPath);
|
|
|
|
|
return $logPath if defined $logPath;
|
2013-08-30 13:53:25 +00:00
|
|
|
|
}
|
2013-10-04 17:01:47 +02:00
|
|
|
|
|
2013-08-30 13:53:25 +00:00
|
|
|
|
return undef if scalar @outPaths == 0;
|
|
|
|
|
|
|
|
|
|
my @steps = $c->model('DB::BuildSteps')->search(
|
2013-08-30 18:11:03 +00:00
|
|
|
|
{ path => { -in => [@outPaths] } },
|
|
|
|
|
{ select => ["drvpath"]
|
2013-08-30 13:53:25 +00:00
|
|
|
|
, distinct => 1
|
2013-08-30 18:11:03 +00:00
|
|
|
|
, join => "buildstepoutputs"
|
|
|
|
|
});
|
2013-08-30 13:53:25 +00:00
|
|
|
|
|
|
|
|
|
foreach my $step (@steps) {
|
2013-09-03 14:43:08 -04:00
|
|
|
|
next unless defined $step->drvpath;
|
2013-08-30 18:11:03 +00:00
|
|
|
|
my $logPath = getDrvLogPath($step->drvpath);
|
|
|
|
|
return $logPath if defined $logPath;
|
2013-08-30 13:53:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return undef;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-06-26 12:00:18 +02:00
|
|
|
|
sub logContents {
|
2013-08-30 13:53:25 +00:00
|
|
|
|
my ($logPath, $tail) = @_;
|
2012-06-26 12:00:18 +02:00
|
|
|
|
my $cmd;
|
2013-01-22 22:48:02 +01:00
|
|
|
|
if ($logPath =~ /.bz2$/) {
|
|
|
|
|
$cmd = "bzip2 -d < $logPath";
|
|
|
|
|
$cmd = $cmd . " | tail -n $tail" if defined $tail;
|
2012-06-26 12:00:18 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
2013-01-22 22:48:02 +01:00
|
|
|
|
$cmd = defined $tail ? "tail -$tail $logPath" : "cat $logPath";
|
2012-06-26 12:00:18 +02:00
|
|
|
|
}
|
2014-08-13 18:53:29 +02:00
|
|
|
|
return decode("utf-8", `$cmd`);
|
2012-06-26 12:00:18 +02:00
|
|
|
|
}
|
2012-03-05 21:52:47 +01:00
|
|
|
|
|
2013-01-22 22:48:02 +01:00
|
|
|
|
|
2011-06-10 09:53:15 +00:00
|
|
|
|
sub removeAsciiEscapes {
|
|
|
|
|
my ($logtext) = @_;
|
2011-06-10 10:53:59 +00:00
|
|
|
|
$logtext =~ s/\e\[[0-9]*[A-Za-z]//g;
|
2011-06-10 09:53:15 +00:00
|
|
|
|
return $logtext;
|
|
|
|
|
}
|
2011-04-18 08:10:10 +00:00
|
|
|
|
|
2012-03-05 21:52:47 +01:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
sub getMainOutput {
|
|
|
|
|
my ($build) = @_;
|
|
|
|
|
return
|
|
|
|
|
$build->buildoutputs->find({name => "out"}) //
|
|
|
|
|
$build->buildoutputs->find({}, {limit => 1, order_by => ["name"]});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-02-26 19:08:39 +01:00
|
|
|
|
sub getEvalInputs {
|
|
|
|
|
my ($c, $eval) = @_;
|
|
|
|
|
my @inputs = $eval->jobsetevalinputs->search(
|
|
|
|
|
{ -or => [ -and => [ uri => { '!=' => undef }, revision => { '!=' => undef }], dependency => { '!=' => undef }], altNr => 0 },
|
|
|
|
|
{ order_by => "name" });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub getEvalInfo {
|
|
|
|
|
my ($cache, $eval) = @_;
|
|
|
|
|
my $res = $cache->{$eval->id}; return $res if defined $res;
|
|
|
|
|
|
|
|
|
|
# Get stats for this eval.
|
|
|
|
|
my $nrScheduled;
|
|
|
|
|
my $nrSucceeded = $eval->nrsucceeded;
|
|
|
|
|
if (defined $nrSucceeded) {
|
|
|
|
|
$nrScheduled = 0;
|
|
|
|
|
} else {
|
|
|
|
|
$nrScheduled = $eval->builds->search({finished => 0})->count;
|
|
|
|
|
$nrSucceeded = $eval->builds->search({finished => 1, buildStatus => 0})->count;
|
|
|
|
|
if ($nrScheduled == 0) {
|
|
|
|
|
$eval->update({nrsucceeded => $nrSucceeded});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Get the inputs.
|
|
|
|
|
my @inputsList = $eval->jobsetevalinputs->search(
|
|
|
|
|
{ -or => [ -and => [ uri => { '!=' => undef }, revision => { '!=' => undef }], dependency => { '!=' => undef }], altNr => 0 },
|
|
|
|
|
{ order_by => "name" });
|
|
|
|
|
my $inputs;
|
|
|
|
|
$inputs->{$_->name} = $_ foreach @inputsList;
|
|
|
|
|
|
|
|
|
|
return $cache->{$eval->id} =
|
|
|
|
|
{ nrScheduled => $nrScheduled
|
|
|
|
|
, nrSucceeded => $nrSucceeded
|
|
|
|
|
, inputs => $inputs
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-02-21 17:27:17 +01:00
|
|
|
|
sub getEvals {
|
|
|
|
|
my ($self, $c, $evals, $offset, $rows) = @_;
|
|
|
|
|
|
|
|
|
|
my @evals = $evals->search(
|
|
|
|
|
{ hasnewbuilds => 1 },
|
2013-02-26 19:08:39 +01:00
|
|
|
|
{ order_by => "id DESC", rows => $rows, offset => $offset });
|
2013-02-21 17:27:17 +01:00
|
|
|
|
|
|
|
|
|
my @res = ();
|
2013-02-26 19:08:39 +01:00
|
|
|
|
my $cache = {};
|
|
|
|
|
|
|
|
|
|
foreach my $curEval (@evals) {
|
|
|
|
|
|
|
|
|
|
my ($prevEval) = $c->model('DB::JobsetEvals')->search(
|
|
|
|
|
{ project => $curEval->get_column('project'), jobset => $curEval->get_column('jobset')
|
|
|
|
|
, hasnewbuilds => 1, id => { '<', $curEval->id } },
|
|
|
|
|
{ order_by => "id DESC", rows => 1 });
|
|
|
|
|
|
|
|
|
|
my $curInfo = getEvalInfo($cache, $curEval);
|
|
|
|
|
my $prevInfo = getEvalInfo($cache, $prevEval) if defined $prevEval;
|
2013-02-21 17:27:17 +01:00
|
|
|
|
|
|
|
|
|
# Compute what inputs changed between each eval.
|
|
|
|
|
my @changedInputs;
|
2015-02-25 13:13:12 +01:00
|
|
|
|
foreach my $input (sort { $a->name cmp $b->name } values(%{$curInfo->{inputs}})) {
|
2013-02-26 19:08:39 +01:00
|
|
|
|
my $p = $prevInfo->{inputs}->{$input->name};
|
|
|
|
|
push @changedInputs, $input if
|
|
|
|
|
!defined $p
|
|
|
|
|
|| ($input->revision || "") ne ($p->revision || "")
|
|
|
|
|
|| $input->type ne $p->type
|
|
|
|
|
|| ($input->uri || "") ne ($p->uri || "")
|
|
|
|
|
|| ($input->get_column('dependency') || "") ne ($p->get_column('dependency') || "");
|
2013-02-21 17:27:17 +01:00
|
|
|
|
}
|
2013-02-26 19:08:39 +01:00
|
|
|
|
|
|
|
|
|
push @res,
|
|
|
|
|
{ eval => $curEval
|
|
|
|
|
, nrScheduled => $curInfo->{nrScheduled}
|
|
|
|
|
, nrSucceeded => $curInfo->{nrSucceeded}
|
|
|
|
|
, nrFailed => $curEval->nrbuilds - $curInfo->{nrSucceeded} - $curInfo->{nrScheduled}
|
|
|
|
|
, diff => defined $prevEval ? $curInfo->{nrSucceeded} - $prevInfo->{nrSucceeded} : 0
|
2013-02-21 17:27:17 +01:00
|
|
|
|
, changedInputs => [ @changedInputs ]
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-26 19:08:39 +01:00
|
|
|
|
return [@res];
|
2013-02-21 17:27:17 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 14:59:36 +01:00
|
|
|
|
|
2013-03-04 15:37:20 -05:00
|
|
|
|
sub getMachines {
|
2013-10-30 11:21:23 +01:00
|
|
|
|
my $machinesConf = $ENV{"NIX_REMOTE_SYSTEMS"} || "/etc/nix/machines";
|
2013-03-04 15:37:20 -05:00
|
|
|
|
|
|
|
|
|
# Read the list of machines.
|
|
|
|
|
my %machines = ();
|
|
|
|
|
if (-e $machinesConf) {
|
|
|
|
|
open CONF, "<$machinesConf" or die;
|
|
|
|
|
while (<CONF>) {
|
|
|
|
|
chomp;
|
|
|
|
|
s/\#.*$//g;
|
|
|
|
|
next if /^\s*$/;
|
|
|
|
|
my @tokens = split /\s/, $_;
|
|
|
|
|
my @supportedFeatures = split(/,/, $tokens[5] || "");
|
|
|
|
|
my @mandatoryFeatures = split(/,/, $tokens[6] || "");
|
|
|
|
|
$machines{$tokens[0]} =
|
|
|
|
|
{ systemTypes => [ split(/,/, $tokens[1]) ]
|
|
|
|
|
, sshKeys => $tokens[2]
|
|
|
|
|
, maxJobs => int($tokens[3])
|
|
|
|
|
, speedFactor => 1.0 * (defined $tokens[4] ? int($tokens[4]) : 1)
|
|
|
|
|
, supportedFeatures => [ @supportedFeatures, @mandatoryFeatures ]
|
|
|
|
|
, mandatoryFeatures => [ @mandatoryFeatures ]
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
close CONF;
|
|
|
|
|
}
|
|
|
|
|
return \%machines;
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-21 17:27:17 +01:00
|
|
|
|
|
2013-04-02 23:32:04 +02:00
|
|
|
|
# Check whether ‘$path’ is inside ‘$prefix’. In particular, it checks
|
|
|
|
|
# that resolving symlink components of ‘$path’ never takes us outside
|
|
|
|
|
# of ‘$prefix’. We use this to check that Nix build products don't
|
|
|
|
|
# refer to things outside of the Nix store (e.g. /etc/passwd) or to
|
|
|
|
|
# symlinks outside of the store that point into the store
|
|
|
|
|
# (e.g. /run/current-system). Return undef or the resolved path.
|
|
|
|
|
sub pathIsInsidePrefix {
|
|
|
|
|
my ($path, $prefix) = @_;
|
|
|
|
|
my $n = 0;
|
|
|
|
|
$path =~ s/\/+/\//g; # remove redundant slashes
|
|
|
|
|
$path =~ s/\/*$//; # remove trailing slashes
|
|
|
|
|
|
|
|
|
|
return undef unless $path eq $prefix || substr($path, 0, length($prefix) + 1) eq "$prefix/";
|
|
|
|
|
|
|
|
|
|
my @cs = File::Spec->splitdir(substr($path, length($prefix) + 1));
|
|
|
|
|
my $cur = $prefix;
|
|
|
|
|
|
|
|
|
|
foreach my $c (@cs) {
|
|
|
|
|
next if $c eq ".";
|
|
|
|
|
|
|
|
|
|
# ‘..’ should not take us outside of the prefix.
|
|
|
|
|
if ($c eq "..") {
|
|
|
|
|
return if length($cur) <= length($prefix);
|
|
|
|
|
$cur =~ s/\/[^\/]*$// or die; # remove last component
|
|
|
|
|
next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
my $new = "$cur/$c";
|
|
|
|
|
if (-l $new) {
|
|
|
|
|
my $link = readlink $new or return undef;
|
|
|
|
|
$new = substr($link, 0, 1) eq "/" ? $link : "$cur/$link";
|
|
|
|
|
$new = pathIsInsidePrefix($new, $prefix);
|
|
|
|
|
return undef unless defined $new;
|
|
|
|
|
}
|
|
|
|
|
$cur = $new;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $cur;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-05-25 15:36:58 -04:00
|
|
|
|
sub captureStdoutStderr {
|
|
|
|
|
my ($timeout, @cmd) = @_;
|
|
|
|
|
my $stdin = "";
|
|
|
|
|
my $stdout;
|
|
|
|
|
my $stderr;
|
|
|
|
|
|
|
|
|
|
eval {
|
|
|
|
|
local $SIG{ALRM} = sub { die "timeout\n" }; # NB: \n required
|
|
|
|
|
alarm $timeout;
|
|
|
|
|
IPC::Run::run(\@cmd, \$stdin, \$stdout, \$stderr);
|
|
|
|
|
alarm 0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
|
die unless $@ eq "timeout\n"; # propagate unexpected errors
|
|
|
|
|
return (-1, "", "timeout\n");
|
|
|
|
|
} else {
|
|
|
|
|
return ($?, $stdout, $stderr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-08-12 17:25:59 +02:00
|
|
|
|
sub run {
|
|
|
|
|
my (%args) = @_;
|
|
|
|
|
my $res = { stdout => "", stderr => "" };
|
|
|
|
|
my $stdin = "";
|
|
|
|
|
|
|
|
|
|
eval {
|
|
|
|
|
local $SIG{ALRM} = sub { die "timeout\n" }; # NB: \n required
|
|
|
|
|
alarm $args{timeout} if defined $args{timeout};
|
|
|
|
|
my @x = ($args{cmd}, \$stdin, \$res->{stdout});
|
|
|
|
|
push @x, \$res->{stderr} if $args{grabStderr} // 1;
|
|
|
|
|
IPC::Run::run(@x,
|
|
|
|
|
init => sub { chdir $args{dir} or die "changing to $args{dir}" if defined $args{dir}; });
|
|
|
|
|
alarm 0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
|
die unless $@ eq "timeout\n"; # propagate unexpected errors
|
|
|
|
|
$res->{status} = -1;
|
|
|
|
|
$res->{stderr} = "timeout\n";
|
|
|
|
|
} else {
|
|
|
|
|
$res->{status} = $?;
|
|
|
|
|
chomp $res->{stdout} if $args{chomp} // 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub grab {
|
|
|
|
|
my (%args) = @_;
|
|
|
|
|
my $res = run(%args, grabStderr => 0);
|
|
|
|
|
die "command `@{$args{cmd}}' failed with exit status $res->{status}" if $res->{status};
|
|
|
|
|
return $res->{stdout};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-21 14:47:52 +00:00
|
|
|
|
sub getTotalShares {
|
|
|
|
|
my ($db) = @_;
|
|
|
|
|
return $db->resultset('Jobsets')->search(
|
2013-10-11 12:01:52 +02:00
|
|
|
|
{ 'project.enabled' => 1, 'me.enabled' => { '!=' => 0 } },
|
2013-09-21 14:47:52 +00:00
|
|
|
|
{ join => 'project', select => { sum => 'schedulingshares' }, as => 'sum' })->single->get_column('sum');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-10-04 15:40:43 +02:00
|
|
|
|
sub cancelBuilds($$) {
|
|
|
|
|
my ($db, $builds) = @_;
|
|
|
|
|
return txn_do($db, sub {
|
|
|
|
|
$builds = $builds->search({ finished => 0, busy => 0 });
|
|
|
|
|
my $n = $builds->count;
|
|
|
|
|
my $time = time();
|
|
|
|
|
$builds->update(
|
|
|
|
|
{ finished => 1,
|
|
|
|
|
, iscachedbuild => 0, buildstatus => 4 # = cancelled
|
|
|
|
|
, starttime => $time
|
|
|
|
|
, stoptime => $time
|
|
|
|
|
});
|
2015-06-11 18:07:45 +02:00
|
|
|
|
$db->storage->dbh->do("notify builds_cancelled");
|
2013-10-04 15:40:43 +02:00
|
|
|
|
return $n;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-10-04 17:01:47 +02:00
|
|
|
|
sub restartBuilds($$) {
|
|
|
|
|
my ($db, $builds) = @_;
|
2014-07-18 00:02:59 +02:00
|
|
|
|
my @buildIds;
|
2013-10-04 17:01:47 +02:00
|
|
|
|
|
|
|
|
|
txn_do($db, sub {
|
|
|
|
|
my @paths;
|
|
|
|
|
|
|
|
|
|
$builds = $builds->search({ finished => 1 });
|
2014-07-18 00:02:59 +02:00
|
|
|
|
|
2013-10-04 17:01:47 +02:00
|
|
|
|
foreach my $build ($builds->all) {
|
|
|
|
|
next if !isValidPath($build->drvpath);
|
|
|
|
|
push @paths, $build->drvpath;
|
2014-07-18 00:02:59 +02:00
|
|
|
|
push @buildIds, $build->id;
|
2013-10-04 17:11:42 +02:00
|
|
|
|
registerRoot $build->drvpath;
|
2013-10-04 17:01:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-18 00:02:59 +02:00
|
|
|
|
$db->resultset('Builds')->search({ id => \@buildIds })->update(
|
|
|
|
|
{ finished => 0
|
|
|
|
|
, busy => 0
|
|
|
|
|
, locker => ""
|
|
|
|
|
, iscachedbuild => 0
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
# Reset the stats for the evals to which the builds belongs.
|
|
|
|
|
# !!! Should do this in a trigger.
|
|
|
|
|
$db->resultset('JobsetEvals')->search({ build => \@buildIds }, { join => 'buildIds' })->update({ nrsucceeded => undef });
|
|
|
|
|
|
2015-06-10 14:57:16 +02:00
|
|
|
|
# Clear the failed paths cache.
|
2013-10-04 17:01:47 +02:00
|
|
|
|
# FIXME: Add this to the API.
|
2015-06-10 14:57:16 +02:00
|
|
|
|
# FIXME: clear the dependencies?
|
|
|
|
|
$db->resultset('FailedPaths')->search({ path => [ @paths ]})->delete;
|
2015-06-11 17:38:55 +02:00
|
|
|
|
|
|
|
|
|
$db->storage->dbh->do("notify builds_restarted");
|
2013-10-04 17:01:47 +02:00
|
|
|
|
});
|
|
|
|
|
|
2014-07-18 00:02:59 +02:00
|
|
|
|
return scalar(@buildIds);
|
2013-10-04 17:01:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-11-28 14:36:04 +00:00
|
|
|
|
1;
|