2009-02-25 12:03:13 +00:00
|
|
|
|
package Hydra::Controller::Build;
|
|
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
|
use warnings;
|
2009-03-04 12:23:54 +00:00
|
|
|
|
use base 'Hydra::Base::Controller::NixChannel';
|
2009-02-25 12:03:13 +00:00
|
|
|
|
use Hydra::Helper::Nix;
|
|
|
|
|
use Hydra::Helper::CatalystUtils;
|
2009-10-26 15:39:14 +00:00
|
|
|
|
use Hydra::Helper::AddBuilds;
|
2009-04-16 15:22:14 +00:00
|
|
|
|
use File::stat;
|
2013-01-22 21:48:02 +00:00
|
|
|
|
use File::Slurp;
|
2010-01-22 13:31:59 +00:00
|
|
|
|
use Data::Dump qw(dump);
|
2011-11-30 14:25:28 +00:00
|
|
|
|
use Nix::Store;
|
2013-02-13 17:34:33 +00:00
|
|
|
|
use Nix::Config;
|
2013-02-13 16:49:28 +00:00
|
|
|
|
use List::MoreUtils qw(all);
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2011-09-15 08:27:17 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
sub build : Chained('/') PathPart CaptureArgs(1) {
|
|
|
|
|
my ($self, $c, $id) = @_;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
$c->stash->{id} = $id;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
$c->stash->{build} = getBuild($c, $id);
|
2010-03-05 13:03:41 +00:00
|
|
|
|
|
|
|
|
|
notFound($c, "Build with ID $id doesn't exist.")
|
|
|
|
|
if !defined $c->stash->{build};
|
|
|
|
|
|
2010-02-22 13:21:34 +00:00
|
|
|
|
$c->stash->{prevBuild} = getPreviousBuild($c, $c->stash->{build});
|
|
|
|
|
$c->stash->{prevSuccessfulBuild} = getPreviousSuccessfulBuild($c, $c->stash->{build});
|
2010-07-14 07:31:14 +00:00
|
|
|
|
$c->stash->{firstBrokenBuild} = getNextBuild($c, $c->stash->{prevSuccessfulBuild});
|
2009-02-25 14:34:29 +00:00
|
|
|
|
|
2010-02-25 15:32:56 +00:00
|
|
|
|
$c->stash->{mappers} = [$c->model('DB::UriRevMapper')->all];
|
|
|
|
|
|
2009-03-13 15:41:19 +00:00
|
|
|
|
$c->stash->{project} = $c->stash->{build}->project;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-09-15 08:27:17 +00:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
sub findBuildStepByOutPath {
|
|
|
|
|
my ($self, $c, $path, $status) = @_;
|
|
|
|
|
return $c->model('DB::BuildSteps')->search(
|
|
|
|
|
{ path => $path, busy => 0, status => $status },
|
|
|
|
|
{ join => ["buildstepoutputs"], order_by => ["stopTime"], limit => 1 })->single;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub findBuildStepByDrvPath {
|
|
|
|
|
my ($self, $c, $drvPath, $status) = @_;
|
|
|
|
|
return $c->model('DB::BuildSteps')->search(
|
|
|
|
|
{ drvpath => $drvPath, busy => 0, status => $status },
|
|
|
|
|
{ order_by => ["stopTime"], limit => 1 })->single;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
sub view_build : Chained('build') PathPart('') Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
$c->stash->{template} = 'build.tt';
|
2013-02-13 16:49:28 +00:00
|
|
|
|
$c->stash->{available} = all { isValidPath($_->path) } $build->buildoutputs->all;
|
2009-02-27 15:31:49 +00:00
|
|
|
|
$c->stash->{drvAvailable} = isValidPath $build->drvpath;
|
2009-03-06 12:49:01 +00:00
|
|
|
|
$c->stash->{flashMsg} = $c->flash->{buildMsg};
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2012-02-29 01:22:49 +00:00
|
|
|
|
if (!$build->finished && $build->busy) {
|
2013-01-22 21:48:02 +00:00
|
|
|
|
$c->stash->{logtext} = read_file($build->logfile, err_mode => 'quiet') // "";
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
2010-01-15 14:18:12 +00:00
|
|
|
|
|
2012-03-05 20:52:47 +00:00
|
|
|
|
if ($build->finished && $build->iscachedbuild) {
|
2013-02-13 16:49:28 +00:00
|
|
|
|
my $path = ($build->buildoutputs)[0]->path or die;
|
|
|
|
|
my $cachedBuildStep = findBuildStepByOutPath($self, $c, $path,
|
|
|
|
|
$build->buildstatus == 0 || $build->buildstatus == 6 ? 0 : 1);
|
2010-01-19 16:35:16 +00:00
|
|
|
|
$c->stash->{cachedBuild} = $cachedBuildStep->build if defined $cachedBuildStep;
|
2010-01-15 14:18:12 +00:00
|
|
|
|
}
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
if ($build->finished && 0) {
|
2012-03-05 20:52:47 +00:00
|
|
|
|
$c->stash->{prevBuilds} = [$c->model('DB::Builds')->search(
|
2010-02-10 10:15:09 +00:00
|
|
|
|
{ project => $c->stash->{project}->name
|
|
|
|
|
, jobset => $c->stash->{build}->jobset->name
|
|
|
|
|
, job => $c->stash->{build}->job->name
|
2012-06-25 13:05:16 +00:00
|
|
|
|
, 'me.system' => $build->system
|
2010-02-10 10:15:09 +00:00
|
|
|
|
, finished => 1
|
|
|
|
|
, buildstatus => 0
|
2010-02-11 12:23:46 +00:00
|
|
|
|
, 'me.id' => { '<=' => $build->id }
|
|
|
|
|
}
|
|
|
|
|
, { join => "actualBuildStep"
|
|
|
|
|
, "+select" => ["actualBuildStep.stoptime - actualBuildStep.starttime"]
|
|
|
|
|
, "+as" => ["actualBuildTime"]
|
2011-03-15 13:54:43 +00:00
|
|
|
|
, order_by => "me.id DESC"
|
2010-02-11 12:23:46 +00:00
|
|
|
|
, rows => 50
|
2010-02-10 10:15:09 +00:00
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-21 17:34:34 +00:00
|
|
|
|
# Get the first eval of which this build was a part.
|
2013-02-21 17:42:18 +00:00
|
|
|
|
($c->stash->{nrEvals}) = $c->stash->{build}->jobsetevals->search({ hasnewbuilds => 1 })->count;
|
2013-02-21 17:34:34 +00:00
|
|
|
|
($c->stash->{eval}) = $c->stash->{build}->jobsetevals->search(
|
|
|
|
|
{ hasnewbuilds => 1},
|
|
|
|
|
{ limit => 1, order_by => ["id"] });
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
|
sub view_nixlog : Chained('build') PathPart('nixlog') {
|
|
|
|
|
my ($self, $c, $stepnr, $mode) = @_;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
|
|
|
|
my $step = $c->stash->{build}->buildsteps->find({stepnr => $stepnr});
|
2009-02-25 14:34:29 +00:00
|
|
|
|
notFound($c, "Build doesn't have a build step $stepnr.") if !defined $step;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
|
|
|
|
$c->stash->{step} = $step;
|
|
|
|
|
|
2013-01-22 21:48:02 +00:00
|
|
|
|
showLog($c, $step->drvpath, $mode);
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
|
sub view_log : Chained('build') PathPart('log') {
|
|
|
|
|
my ($self, $c, $mode) = @_;
|
2013-01-22 21:48:02 +00:00
|
|
|
|
showLog($c, $c->stash->{build}->drvpath, $mode);
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
|
sub showLog {
|
2013-01-22 21:48:02 +00:00
|
|
|
|
my ($c, $drvPath, $mode) = @_;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2013-01-22 21:48:02 +00:00
|
|
|
|
my $logPath = getDrvLogPath($drvPath);
|
2010-08-10 13:11:41 +00:00
|
|
|
|
|
2013-01-22 21:48:02 +00:00
|
|
|
|
notFound($c, "The build log of derivation ‘$drvPath’ is not available.") unless defined $logPath;
|
2010-08-10 13:11:41 +00:00
|
|
|
|
|
2009-03-16 12:16:33 +00:00
|
|
|
|
if (!$mode) {
|
2009-03-14 23:27:08 +00:00
|
|
|
|
# !!! quick hack
|
2013-01-22 21:48:02 +00:00
|
|
|
|
my $pipeline = "nix-store -l $drvPath"
|
2009-03-14 23:27:08 +00:00
|
|
|
|
. " | nix-log2xml | xsltproc " . $c->path_to("xsl/mark-errors.xsl") . " -"
|
|
|
|
|
. " | xsltproc " . $c->path_to("xsl/log2html.xsl") . " - | tail -n +2";
|
|
|
|
|
$c->stash->{template} = 'log.tt';
|
|
|
|
|
$c->stash->{logtext} = `$pipeline`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elsif ($mode eq "raw") {
|
2013-01-22 21:48:02 +00:00
|
|
|
|
$c->stash->{'plain'} = { data => (scalar logContents($drvPath)) || " " };
|
2010-08-10 13:11:41 +00:00
|
|
|
|
$c->forward('Hydra::View::Plain');
|
2009-03-14 23:27:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-02-02 09:00:52 +00:00
|
|
|
|
elsif ($mode eq "tail-reload") {
|
2013-01-22 13:09:37 +00:00
|
|
|
|
my $url = $c->request->uri->as_string;
|
|
|
|
|
$url =~ s/tail-reload/tail/g;
|
2011-02-02 09:00:52 +00:00
|
|
|
|
$c->stash->{url} = $url;
|
2012-02-29 01:22:49 +00:00
|
|
|
|
$c->stash->{reload} = !$c->stash->{build}->finished && $c->stash->{build}->busy;
|
2011-02-02 09:00:52 +00:00
|
|
|
|
$c->stash->{title} = "";
|
2013-01-22 21:48:02 +00:00
|
|
|
|
$c->stash->{contents} = (scalar logContents($drvPath, 50)) || " ";
|
2011-02-02 09:00:52 +00:00
|
|
|
|
$c->stash->{template} = 'plain-reload.tt';
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-16 12:16:33 +00:00
|
|
|
|
elsif ($mode eq "tail") {
|
2013-01-22 21:48:02 +00:00
|
|
|
|
$c->stash->{'plain'} = { data => (scalar logContents($drvPath, 50)) || " " };
|
2009-03-16 12:16:33 +00:00
|
|
|
|
$c->forward('Hydra::View::Plain');
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
|
else {
|
|
|
|
|
error($c, "Unknown log display mode `$mode'.");
|
|
|
|
|
}
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-04-08 08:09:39 +00:00
|
|
|
|
sub defaultUriForProduct {
|
|
|
|
|
my ($self, $c, $product, @path) = @_;
|
|
|
|
|
my $x = $product->productnr
|
|
|
|
|
. ($product->name ? "/" . $product->name : "")
|
2012-06-25 13:05:16 +00:00
|
|
|
|
. ($product->defaultpath ? "/" . $product->defaultpath : "");
|
2011-08-19 16:23:01 +00:00
|
|
|
|
return $c->uri_for($self->action_for("download"), $c->req->captures, (split /\//, $x), @path);
|
2009-04-08 08:09:39 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-02-13 17:34:33 +00:00
|
|
|
|
sub checkPath {
|
|
|
|
|
my ($self, $c, $path) = @_;
|
|
|
|
|
my $storeDir = $Nix::Config::storeDir . "/";
|
|
|
|
|
error($c, "Invalid path in build product.")
|
|
|
|
|
if substr($path, 0, length($storeDir)) ne $storeDir || $path =~ /\/\.\./;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-18 17:40:12 +00:00
|
|
|
|
sub download : Chained('build') PathPart {
|
2009-03-06 13:34:53 +00:00
|
|
|
|
my ($self, $c, $productnr, @path) = @_;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2009-04-08 09:37:05 +00:00
|
|
|
|
$productnr = 1 if !defined $productnr;
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
my $product = $c->stash->{build}->buildproducts->find({productnr => $productnr});
|
2009-04-08 09:37:05 +00:00
|
|
|
|
notFound($c, "Build doesn't have a product #$productnr.") if !defined $product;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2013-02-13 17:34:33 +00:00
|
|
|
|
notFound($c, "Build product " . $product->path . " has disappeared.") unless -e $product->path;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2009-04-08 08:09:39 +00:00
|
|
|
|
return $c->res->redirect(defaultUriForProduct($self, $c, $product, @path))
|
|
|
|
|
if scalar @path == 0 && ($product->name || $product->defaultpath);
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-03-06 13:34:53 +00:00
|
|
|
|
# If the product has a name, then the first path element can be
|
|
|
|
|
# ignored (it's the name included in the URL for informational purposes).
|
2012-06-25 13:05:16 +00:00
|
|
|
|
shift @path if $product->name;
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
# Security paranoia.
|
|
|
|
|
foreach my $elem (@path) {
|
2009-02-25 14:34:29 +00:00
|
|
|
|
error($c, "Invalid filename $elem.") if $elem !~ /^$pathCompRE$/;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
my $path = $product->path;
|
|
|
|
|
$path .= "/" . join("/", @path) if scalar @path > 0;
|
|
|
|
|
|
2013-02-13 17:34:33 +00:00
|
|
|
|
# Make sure the file is in the Nix store.
|
|
|
|
|
checkPath($self, $c, $path);
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
# If this is a directory but no "/" is attached, then redirect.
|
|
|
|
|
if (-d $path && substr($c->request->uri, -1) ne "/") {
|
|
|
|
|
return $c->res->redirect($c->request->uri . "/");
|
|
|
|
|
}
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
$path = "$path/index.html" if -d $path && -e "$path/index.html";
|
|
|
|
|
|
2009-02-25 14:34:29 +00:00
|
|
|
|
notFound($c, "File $path does not exist.") if !-e $path;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
2009-03-06 13:34:53 +00:00
|
|
|
|
notFound($c, "Path $path is a directory.") if -d $path;
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
$c->serve_static_file($path);
|
2011-09-15 15:22:00 +00:00
|
|
|
|
$c->response->headers->last_modified($c->stash->{build}->timestamp);
|
2009-02-25 12:03:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-04-08 08:09:39 +00:00
|
|
|
|
# Redirect to a download with the given type. Useful when you want to
|
|
|
|
|
# link to some build product of the latest build (i.e. in conjunction
|
|
|
|
|
# with the .../latest redirect).
|
|
|
|
|
sub download_by_type : Chained('build') PathPart('download-by-type') {
|
|
|
|
|
my ($self, $c, $type, $subtype, @path) = @_;
|
|
|
|
|
|
|
|
|
|
notFound($c, "You need to specify a type and a subtype in the URI.")
|
|
|
|
|
unless defined $type && defined $subtype;
|
|
|
|
|
|
|
|
|
|
(my $product) = $c->stash->{build}->buildproducts->search(
|
2009-10-12 17:07:36 +00:00
|
|
|
|
{type => $type, subtype => $subtype}, {order_by => "productnr"});
|
2009-04-08 08:09:39 +00:00
|
|
|
|
notFound($c, "Build doesn't have a build product with type $type/$subtype.")
|
|
|
|
|
if !defined $product;
|
|
|
|
|
|
|
|
|
|
$c->res->redirect(defaultUriForProduct($self, $c, $product, @path));
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-15 08:27:17 +00:00
|
|
|
|
|
2009-03-18 18:50:42 +00:00
|
|
|
|
sub contents : Chained('build') PathPart Args(1) {
|
|
|
|
|
my ($self, $c, $productnr) = @_;
|
2009-03-18 17:40:12 +00:00
|
|
|
|
|
|
|
|
|
my $product = $c->stash->{build}->buildproducts->find({productnr => $productnr});
|
|
|
|
|
notFound($c, "Build doesn't have a product $productnr.") if !defined $product;
|
|
|
|
|
|
|
|
|
|
my $path = $product->path;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2013-02-13 17:34:33 +00:00
|
|
|
|
checkPath($self, $c, $path);
|
|
|
|
|
|
2009-03-18 17:40:12 +00:00
|
|
|
|
notFound($c, "Product $path has disappeared.") unless -e $path;
|
|
|
|
|
|
|
|
|
|
my $res;
|
|
|
|
|
|
2010-02-05 20:07:49 +00:00
|
|
|
|
if ($product->type eq "nix-build" && -d $path) {
|
2009-03-18 17:40:12 +00:00
|
|
|
|
$res = `cd $path && find . -print0 | xargs -0 ls -ld --`;
|
|
|
|
|
error($c, "`ls -lR' error: $?") if $? != 0;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2011-01-04 12:50:59 +00:00
|
|
|
|
my $baseuri = $c->uri_for('/build', $c->stash->{build}->id, 'download', $product->productnr);
|
|
|
|
|
$baseuri .= "/".$product->name if $product->name;
|
|
|
|
|
$res =~ s/(\.\/)($relPathRE)/<a href="$baseuri\/$2">$1$2<\/a>/g;
|
2009-03-18 17:40:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elsif ($path =~ /\.rpm$/) {
|
|
|
|
|
$res = `rpm --query --info --package "$path"`;
|
|
|
|
|
error($c, "RPM error: $?") if $? != 0;
|
|
|
|
|
$res .= "===\n";
|
|
|
|
|
$res .= `rpm --query --list --verbose --package "$path"`;
|
|
|
|
|
error($c, "RPM error: $?") if $? != 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elsif ($path =~ /\.deb$/) {
|
|
|
|
|
$res = `dpkg-deb --info "$path"`;
|
|
|
|
|
error($c, "`dpkg-deb' error: $?") if $? != 0;
|
|
|
|
|
$res .= "===\n";
|
|
|
|
|
$res .= `dpkg-deb --contents "$path"`;
|
|
|
|
|
error($c, "`dpkg-deb' error: $?") if $? != 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-16 15:32:32 +00:00
|
|
|
|
elsif ($path =~ /\.(tar(\.gz|\.bz2|\.xz|\.lzma)?|tgz)$/ ) {
|
2009-03-18 17:40:12 +00:00
|
|
|
|
$res = `tar tvfa "$path"`;
|
|
|
|
|
error($c, "`tar' error: $?") if $? != 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-06-18 13:23:04 +00:00
|
|
|
|
elsif ($path =~ /\.(zip|jar)$/ ) {
|
2009-03-18 17:40:12 +00:00
|
|
|
|
$res = `unzip -v "$path"`;
|
|
|
|
|
error($c, "`unzip' error: $?") if $? != 0;
|
2009-07-07 11:37:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elsif ($path =~ /\.iso$/ ) {
|
|
|
|
|
$res = `isoinfo -d -i "$path" && isoinfo -l -R -i "$path"`;
|
|
|
|
|
error($c, "`isoinfo' error: $?") if $? != 0;
|
2009-03-18 17:40:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else {
|
|
|
|
|
error($c, "Unsupported file type.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
die unless $res;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2011-01-04 12:50:59 +00:00
|
|
|
|
$c->stash->{title} = "Contents of ".$product->path;
|
|
|
|
|
$c->stash->{contents} = "<pre>$res</pre>";
|
|
|
|
|
$c->stash->{template} = 'plain.tt';
|
2009-03-18 17:40:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-02-14 14:53:34 +00:00
|
|
|
|
sub getDependencyGraph {
|
|
|
|
|
my ($self, $c, $runtime, $done, $path) = @_;
|
|
|
|
|
my $node = $$done{$path};
|
|
|
|
|
|
|
|
|
|
if (!defined $node) {
|
|
|
|
|
$path =~ /\/[a-z0-9]+-(.*)$/;
|
|
|
|
|
my $name = $1 // $path;
|
|
|
|
|
$name =~ s/\.drv$//;
|
|
|
|
|
$node =
|
|
|
|
|
{ path => $path
|
|
|
|
|
, name => $name
|
|
|
|
|
, buildStep => $runtime
|
|
|
|
|
? findBuildStepByOutPath($self, $c, $path, 0)
|
|
|
|
|
: findBuildStepByDrvPath($self, $c, $path, 0)
|
|
|
|
|
};
|
|
|
|
|
$$done{$path} = $node;
|
|
|
|
|
my @refs;
|
|
|
|
|
foreach my $ref (queryReferences($path)) {
|
|
|
|
|
next if $ref eq $path;
|
|
|
|
|
next unless $runtime || $ref =~ /\.drv$/;
|
|
|
|
|
getDependencyGraph($self, $c, $runtime, $done, $ref);
|
|
|
|
|
push @refs, $ref;
|
|
|
|
|
}
|
|
|
|
|
# Show in reverse topological order to flatten the graph.
|
|
|
|
|
# Should probably do a proper BFS.
|
|
|
|
|
my @sorted = reverse topoSortPaths(@refs);
|
|
|
|
|
$node->{refs} = [map { $$done{$_} } @sorted];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $node;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-02-20 16:58:27 +00:00
|
|
|
|
sub build_deps : Chained('build') PathPart('build-deps') {
|
2010-01-22 13:31:59 +00:00
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
my $build = $c->stash->{build};
|
2013-02-13 16:49:28 +00:00
|
|
|
|
my $drvPath = $build->drvpath;
|
2010-01-22 13:31:59 +00:00
|
|
|
|
|
2013-02-20 16:58:27 +00:00
|
|
|
|
error($c, "Derivation no longer available.") unless isValidPath $drvPath;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2013-02-20 16:58:27 +00:00
|
|
|
|
$c->stash->{buildTimeGraph} = getDependencyGraph($self, $c, 0, {}, $drvPath);
|
2010-01-22 13:31:59 +00:00
|
|
|
|
|
2013-02-20 16:58:27 +00:00
|
|
|
|
$c->stash->{template} = 'build-deps.tt';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub runtime_deps : Chained('build') PathPart('runtime-deps') {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
my @outPaths = map { $_->path } $build->buildoutputs->all;
|
|
|
|
|
|
|
|
|
|
error($c, "Build outputs no longer available.") unless all { isValidPath($_) } @outPaths;
|
|
|
|
|
|
|
|
|
|
my $done = {};
|
|
|
|
|
$c->stash->{runtimeGraph} = [ map { getDependencyGraph($self, $c, 1, $done, $_) } @outPaths ];
|
2010-01-22 13:31:59 +00:00
|
|
|
|
|
2013-02-20 16:58:27 +00:00
|
|
|
|
$c->stash->{template} = 'runtime-deps.tt';
|
2010-01-22 13:31:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-02-27 15:31:49 +00:00
|
|
|
|
|
2009-02-25 14:34:29 +00:00
|
|
|
|
sub nix : Chained('build') PathPart('nix') CaptureArgs(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
2009-02-27 14:57:06 +00:00
|
|
|
|
notFound($c, "Build cannot be downloaded as a closure or Nix package.")
|
2013-02-13 16:49:28 +00:00
|
|
|
|
if $build->buildproducts->search({type => "nix-build"})->count == 0;
|
2009-02-25 14:34:29 +00:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
foreach my $out ($build->buildoutputs) {
|
|
|
|
|
notFound($c, "Path " . $out->path . " is no longer available.")
|
|
|
|
|
unless isValidPath($out->path);
|
|
|
|
|
}
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
$c->stash->{channelBuilds} = $c->model('DB::Builds')->search(
|
|
|
|
|
{ id => $build->id },
|
|
|
|
|
{ join => ["buildoutputs"]
|
|
|
|
|
, '+select' => ['buildoutputs.path', 'buildoutputs.name'], '+as' => ['outpath', 'outname'] });
|
2009-02-25 14:34:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-06 12:49:01 +00:00
|
|
|
|
sub restart : Chained('build') PathPart Args(0) {
|
2009-03-02 16:03:41 +00:00
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
|
|
requireProjectOwner($c, $build->project);
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2012-04-17 09:34:35 +00:00
|
|
|
|
my $drvpath = $build->drvpath;
|
2010-12-03 08:40:34 +00:00
|
|
|
|
error($c, "This build cannot be restarted.")
|
2012-04-17 09:34:35 +00:00
|
|
|
|
unless $build->finished && -f $drvpath;
|
2009-03-02 16:03:41 +00:00
|
|
|
|
|
2010-12-07 13:25:29 +00:00
|
|
|
|
restartBuild($c->model('DB')->schema, $build);
|
2009-03-02 16:03:41 +00:00
|
|
|
|
|
2009-03-06 12:49:01 +00:00
|
|
|
|
$c->flash->{buildMsg} = "Build has been restarted.";
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-03-06 12:49:01 +00:00
|
|
|
|
$c->res->redirect($c->uri_for($self->action_for("view_build"), $c->req->captures));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub cancel : Chained('build') PathPart Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
2009-04-22 22:43:04 +00:00
|
|
|
|
txn_do($c->model('DB')->schema, sub {
|
2009-03-06 12:49:01 +00:00
|
|
|
|
error($c, "This build cannot be cancelled.")
|
2012-02-29 01:22:49 +00:00
|
|
|
|
if $build->finished || $build->busy;
|
2009-03-06 12:49:01 +00:00
|
|
|
|
|
|
|
|
|
# !!! Actually, it would be nice to be able to cancel busy
|
|
|
|
|
# builds as well, but we would have to send a signal or
|
|
|
|
|
# something to the build process.
|
|
|
|
|
|
2012-03-08 00:17:11 +00:00
|
|
|
|
$build->update(
|
|
|
|
|
{ finished => 1, busy => 0, timestamp => time
|
|
|
|
|
, iscachedbuild => 0, buildstatus => 4 # = cancelled
|
2009-03-06 12:49:01 +00:00
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$c->flash->{buildMsg} = "Build has been cancelled.";
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-03-04 10:59:14 +00:00
|
|
|
|
$c->res->redirect($c->uri_for($self->action_for("view_build"), $c->req->captures));
|
2009-03-02 16:03:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-14 23:56:57 +00:00
|
|
|
|
sub keep : Chained('build') PathPart Args(1) {
|
2013-02-13 16:49:28 +00:00
|
|
|
|
my ($self, $c, $x) = @_;
|
|
|
|
|
my $keep = $x eq "1" ? 1 : 0;
|
2009-03-14 23:56:57 +00:00
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
if ($keep) {
|
|
|
|
|
registerRoot $_->path foreach $build->buildoutputs;
|
|
|
|
|
}
|
2009-03-16 10:57:44 +00:00
|
|
|
|
|
2009-04-22 22:43:04 +00:00
|
|
|
|
txn_do($c->model('DB')->schema, sub {
|
2013-02-13 16:49:28 +00:00
|
|
|
|
$build->update({keep => $keep});
|
2009-03-14 23:56:57 +00:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$c->flash->{buildMsg} =
|
2013-02-13 16:49:28 +00:00
|
|
|
|
$keep ? "Build will be kept." : "Build will not be kept.";
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-03-14 23:56:57 +00:00
|
|
|
|
$c->res->redirect($c->uri_for($self->action_for("view_build"), $c->req->captures));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-10-23 15:57:18 +00:00
|
|
|
|
sub add_to_release : Chained('build') PathPart('add-to-release') Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-23 15:57:18 +00:00
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
|
|
|
|
my $releaseName = trim $c->request->params->{name};
|
|
|
|
|
|
|
|
|
|
my $release = $build->project->releases->find({name => $releaseName});
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-23 15:57:18 +00:00
|
|
|
|
error($c, "This project has no release named `$releaseName'.") unless $release;
|
|
|
|
|
|
|
|
|
|
error($c, "This build is already a part of release `$releaseName'.")
|
|
|
|
|
if $release->releasemembers->find({build => $build->id});
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2013-02-13 16:49:28 +00:00
|
|
|
|
foreach my $output ($build->buildoutputs) {
|
|
|
|
|
error($c, "This build is no longer available.") unless isValidPath $output->path;
|
|
|
|
|
registerRoot $output->path;
|
|
|
|
|
}
|
2009-10-23 15:57:18 +00:00
|
|
|
|
|
|
|
|
|
$release->releasemembers->create({build => $build->id, description => $build->description});
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-23 15:57:18 +00:00
|
|
|
|
$c->flash->{buildMsg} = "Build added to project <tt>$releaseName</tt>.";
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-23 15:57:18 +00:00
|
|
|
|
$c->res->redirect($c->uri_for($self->action_for("view_build"), $c->req->captures));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-10-26 14:30:42 +00:00
|
|
|
|
sub clone : Chained('build') PathPart('clone') Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-26 14:30:42 +00:00
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
|
|
|
|
$c->stash->{template} = 'clone-build.tt';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub clone_submit : Chained('build') PathPart('clone/submit') Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-26 14:30:42 +00:00
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
2009-10-26 17:01:23 +00:00
|
|
|
|
my ($nixExprPath, $nixExprInputName) = Hydra::Controller::Jobset::nixExprPathFromParams $c;
|
2009-10-26 15:39:14 +00:00
|
|
|
|
|
2012-08-16 17:07:04 +00:00
|
|
|
|
# When the expression is in a .scm file, assume it's a Guile + Guix
|
|
|
|
|
# build expression.
|
|
|
|
|
my $exprType =
|
2013-01-22 13:09:37 +00:00
|
|
|
|
$c->request->params->{"nixexprpath"} =~ /.scm$/ ? "guile" : "nix";
|
2012-08-16 17:07:04 +00:00
|
|
|
|
|
2009-10-26 15:39:14 +00:00
|
|
|
|
my $jobName = trim $c->request->params->{"jobname"};
|
|
|
|
|
error($c, "Invalid job name: $jobName") if $jobName !~ /^$jobNameRE$/;
|
|
|
|
|
|
2009-10-26 17:01:23 +00:00
|
|
|
|
my $inputInfo = {};
|
|
|
|
|
|
2009-10-26 15:39:14 +00:00
|
|
|
|
foreach my $param (keys %{$c->request->params}) {
|
|
|
|
|
next unless $param =~ /^input-(\w+)-name$/;
|
|
|
|
|
my $baseName = $1;
|
|
|
|
|
my ($inputName, $inputType) =
|
|
|
|
|
Hydra::Controller::Jobset::checkInput($c, $baseName);
|
|
|
|
|
my $inputValue = Hydra::Controller::Jobset::checkInputValue(
|
|
|
|
|
$c, $inputType, $c->request->params->{"input-$baseName-value"});
|
|
|
|
|
eval {
|
2009-10-26 17:01:23 +00:00
|
|
|
|
# !!! fetchInput can take a long time, which might cause
|
|
|
|
|
# the current HTTP request to time out. So maybe this
|
|
|
|
|
# should be done asynchronously. But then error reporting
|
|
|
|
|
# becomes harder.
|
|
|
|
|
my $info = fetchInput(
|
2009-10-26 15:39:14 +00:00
|
|
|
|
$c->model('DB'), $build->project, $build->jobset,
|
|
|
|
|
$inputName, $inputType, $inputValue);
|
2009-10-26 17:01:23 +00:00
|
|
|
|
push @{$$inputInfo{$inputName}}, $info if defined $info;
|
2009-10-26 15:39:14 +00:00
|
|
|
|
};
|
|
|
|
|
error($c, $@) if $@;
|
|
|
|
|
}
|
|
|
|
|
|
2012-08-16 17:07:04 +00:00
|
|
|
|
my ($jobs, $nixExprInput) = evalJobs($inputInfo, $exprType, $nixExprInputName, $nixExprPath);
|
2009-10-26 17:01:23 +00:00
|
|
|
|
|
|
|
|
|
my $job;
|
|
|
|
|
foreach my $j (@{$jobs->{job}}) {
|
|
|
|
|
print STDERR $j->{jobName}, "\n";
|
|
|
|
|
if ($j->{jobName} eq $jobName) {
|
|
|
|
|
error($c, "Nix expression returned multiple builds for job $jobName.")
|
|
|
|
|
if $job;
|
|
|
|
|
$job = $j;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
error($c, "Nix expression did not return a job named $jobName.") unless $job;
|
|
|
|
|
|
|
|
|
|
my %currentBuilds;
|
|
|
|
|
my $newBuild = checkBuild(
|
|
|
|
|
$c->model('DB'), $build->project, $build->jobset,
|
2012-04-02 15:56:29 +00:00
|
|
|
|
$inputInfo, $nixExprInput, $job, \%currentBuilds, undef, {});
|
2009-10-26 17:01:23 +00:00
|
|
|
|
|
|
|
|
|
error($c, "This build has already been performed.") unless $newBuild;
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-26 17:01:23 +00:00
|
|
|
|
$c->flash->{buildMsg} = "Build " . $newBuild->id . " added to the queue.";
|
2012-06-25 13:05:16 +00:00
|
|
|
|
|
2009-10-26 14:30:42 +00:00
|
|
|
|
$c->res->redirect($c->uri_for($c->controller('Root')->action_for('queue')));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2011-09-15 08:27:17 +00:00
|
|
|
|
sub get_info : Chained('build') PathPart('api/get-info') Args(0) {
|
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
# !!! strip the json prefix
|
|
|
|
|
$c->stash->{jsonBuildId} = $build->id;
|
|
|
|
|
$c->stash->{jsonDrvPath} = $build->drvpath;
|
2013-02-13 17:36:01 +00:00
|
|
|
|
my $out = getMainOutput($build);
|
2013-02-13 16:49:28 +00:00
|
|
|
|
$c->stash->{jsonOutPath} = $out->path if defined $out;
|
2011-09-15 08:27:17 +00:00
|
|
|
|
$c->forward('View::JSON');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
|
1;
|