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-04-16 15:22:14 +00:00
|
|
|
use File::stat;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
sub build : Chained('/') PathPart CaptureArgs(1) {
|
|
|
|
my ($self, $c, $id) = @_;
|
|
|
|
|
|
|
|
$c->stash->{id} = $id;
|
|
|
|
|
|
|
|
$c->stash->{build} = getBuild($c, $id);
|
2009-02-25 14:34:29 +00:00
|
|
|
|
|
|
|
notFound($c, "Build with ID $id doesn't exist.")
|
|
|
|
if !defined $c->stash->{build};
|
2009-02-25 12:03:13 +00:00
|
|
|
|
2009-03-13 15:41:19 +00:00
|
|
|
$c->stash->{project} = $c->stash->{build}->project;
|
2009-02-25 12:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub view_build : Chained('build') PathPart('') Args(0) {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
$c->stash->{template} = 'build.tt';
|
|
|
|
$c->stash->{curTime} = time;
|
|
|
|
$c->stash->{available} = isValidPath $build->outpath;
|
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
|
|
|
|
|
|
|
if (!$build->finished && $build->schedulingInfo->busy) {
|
|
|
|
my $logfile = $build->schedulingInfo->logfile;
|
2009-03-06 14:03:09 +00:00
|
|
|
$c->stash->{logtext} = `cat $logfile` if -e $logfile;
|
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;
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
showLog($c, $step->logfile, $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) = @_;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
2009-02-25 14:34:29 +00:00
|
|
|
error($c, "Build didn't produce a log.") if !defined $c->stash->{build}->resultInfo->logfile;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
showLog($c, $c->stash->{build}->resultInfo->logfile, $mode);
|
2009-02-25 12:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-14 23:27:08 +00:00
|
|
|
sub showLog {
|
|
|
|
my ($c, $path, $mode) = @_;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
2009-03-03 22:20:24 +00:00
|
|
|
notFound($c, "Log file $path no longer exists.") unless -f $path;
|
2009-02-25 12:03:13 +00:00
|
|
|
|
2009-04-16 15:22:14 +00:00
|
|
|
# Don't do pretty printing for large logs, because the XSLT
|
|
|
|
# processing is rather slow.
|
|
|
|
if (!$mode && stat($path)->size >= (256 * 1024)) {
|
|
|
|
$mode = "raw";
|
|
|
|
}
|
|
|
|
|
2009-03-16 12:16:33 +00:00
|
|
|
if (!$mode) {
|
2009-03-14 23:27:08 +00:00
|
|
|
# !!! quick hack
|
|
|
|
my $pipeline = ($path =~ /.bz2$/ ? "cat $path | bzip2 -d" : "cat $path")
|
|
|
|
. " | 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") {
|
|
|
|
$c->serve_static_file($path);
|
|
|
|
}
|
|
|
|
|
2009-03-16 12:16:33 +00:00
|
|
|
elsif ($mode eq "tail") {
|
2009-03-20 10:26:50 +00:00
|
|
|
$c->stash->{'plain'} = { data => (scalar `tail -n 50 $path`) || " " };
|
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 : "")
|
|
|
|
. ($product->defaultpath ? "/" . $product->defaultpath : "");
|
|
|
|
return $c->uri_for($self->action_for("download"), $c->req->captures, $x, @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
|
|
|
|
2009-02-27 15:31:49 +00:00
|
|
|
notFound($c, "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);
|
|
|
|
|
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).
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
my $path = $product->path;
|
|
|
|
$path .= "/" . join("/", @path) if scalar @path > 0;
|
|
|
|
|
|
|
|
# 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 . "/");
|
|
|
|
}
|
|
|
|
|
|
|
|
$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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
notFound($c, "Product $path has disappeared.") unless -e $path;
|
|
|
|
|
|
|
|
my $res;
|
|
|
|
|
|
|
|
if ($product->type eq "nix-build") {
|
|
|
|
$res = `cd $path && find . -print0 | xargs -0 ls -ld --`;
|
|
|
|
error($c, "`ls -lR' error: $?") if $? != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
elsif ($path =~ /\.tar(\.gz|\.bz2|\.lzma)?$/ ) {
|
|
|
|
$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;
|
|
|
|
|
|
|
|
$c->stash->{'plain'} = { data => $res };
|
|
|
|
$c->forward('Hydra::View::Plain');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-02-27 15:31:49 +00:00
|
|
|
sub runtimedeps : Chained('build') PathPart('runtime-deps') {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
notFound($c, "Path " . $build->outpath . " is no longer available.")
|
|
|
|
unless isValidPath($build->outpath);
|
|
|
|
|
|
|
|
$c->stash->{current_view} = 'Hydra::View::NixDepGraph';
|
|
|
|
$c->stash->{storePaths} = [$build->outpath];
|
|
|
|
|
2009-03-04 10:59:14 +00:00
|
|
|
$c->res->content_type('image/png'); # !!!
|
2009-02-27 15:31:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub buildtimedeps : Chained('build') PathPart('buildtime-deps') {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
notFound($c, "Path " . $build->drvpath . " is no longer available.")
|
|
|
|
unless isValidPath($build->drvpath);
|
|
|
|
|
|
|
|
$c->stash->{current_view} = 'Hydra::View::NixDepGraph';
|
|
|
|
$c->stash->{storePaths} = [$build->drvpath];
|
|
|
|
|
2009-03-04 10:59:14 +00:00
|
|
|
$c->res->content_type('image/png'); # !!!
|
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.")
|
2009-02-25 14:34:29 +00:00
|
|
|
if !$build->buildproducts->find({type => "nix-build"});
|
|
|
|
|
2009-02-27 14:57:06 +00:00
|
|
|
notFound($c, "Path " . $build->outpath . " is no longer available.")
|
2009-02-25 14:34:29 +00:00
|
|
|
unless isValidPath($build->outpath);
|
|
|
|
|
|
|
|
$c->stash->{storePaths} = [$build->outpath];
|
2009-02-25 16:29:54 +00:00
|
|
|
|
2009-03-04 14:49:21 +00:00
|
|
|
my $pkgName = $build->nixname . "-" . $build->system;
|
|
|
|
$c->stash->{nixPkgs} = {"${pkgName}.nixpkg" => {build => $build, name => $pkgName}};
|
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);
|
|
|
|
|
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 restarted.")
|
|
|
|
unless $build->finished &&
|
2009-06-18 13:23:04 +00:00
|
|
|
($build->resultInfo->buildstatus == 2 ||
|
|
|
|
$build->resultInfo->buildstatus == 3 ||
|
2009-03-06 12:49:01 +00:00
|
|
|
$build->resultInfo->buildstatus == 4);
|
|
|
|
|
2009-03-09 16:22:41 +00:00
|
|
|
$build->update({finished => 0, timestamp => time});
|
2009-03-02 16:03:41 +00:00
|
|
|
|
|
|
|
$build->resultInfo->delete;
|
|
|
|
|
|
|
|
$c->model('DB::BuildSchedulingInfo')->create(
|
|
|
|
{ id => $build->id
|
|
|
|
, priority => 0 # don't know the original priority anymore...
|
|
|
|
, busy => 0
|
|
|
|
, locker => ""
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2009-03-06 12:49:01 +00:00
|
|
|
$c->flash->{buildMsg} = "Build has been restarted.";
|
|
|
|
|
|
|
|
$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.")
|
|
|
|
if $build->finished || $build->schedulingInfo->busy;
|
|
|
|
|
|
|
|
# !!! 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.
|
|
|
|
|
2009-03-14 23:56:57 +00:00
|
|
|
$build->update({finished => 1, timestamp => time});
|
2009-03-06 12:49:01 +00:00
|
|
|
|
|
|
|
$c->model('DB::BuildResultInfo')->create(
|
|
|
|
{ id => $build->id
|
|
|
|
, iscachedbuild => 0
|
|
|
|
, buildstatus => 4 # = cancelled
|
|
|
|
});
|
|
|
|
|
|
|
|
$build->schedulingInfo->delete;
|
|
|
|
});
|
|
|
|
|
|
|
|
$c->flash->{buildMsg} = "Build has been cancelled.";
|
2009-03-02 16:03:41 +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) {
|
|
|
|
my ($self, $c, $newStatus) = @_;
|
|
|
|
|
|
|
|
my $build = $c->stash->{build};
|
|
|
|
|
|
|
|
requireProjectOwner($c, $build->project);
|
|
|
|
|
|
|
|
die unless $newStatus == 0 || $newStatus == 1;
|
|
|
|
|
2009-03-16 10:57:44 +00:00
|
|
|
registerRoot $build->outpath if $newStatus == 1;
|
|
|
|
|
2009-04-22 22:43:04 +00:00
|
|
|
txn_do($c->model('DB')->schema, sub {
|
2009-03-14 23:56:57 +00:00
|
|
|
$build->resultInfo->update({keep => int $newStatus});
|
|
|
|
});
|
|
|
|
|
|
|
|
$c->flash->{buildMsg} =
|
|
|
|
$newStatus == 0 ? "Build will not be kept." : "Build will be kept.";
|
|
|
|
|
|
|
|
$c->res->redirect($c->uri_for($self->action_for("view_build"), $c->req->captures));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-02-25 12:03:13 +00:00
|
|
|
1;
|