2008-11-25 11:01:42 +00:00
|
|
|
package Hydra::Controller::Root;
|
2008-10-28 10:19:31 +00:00
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
2009-03-04 10:59:14 +00:00
|
|
|
use base 'Hydra::Base::Controller::ListBuilds';
|
2008-11-25 11:01:42 +00:00
|
|
|
use Hydra::Helper::Nix;
|
2009-02-25 12:03:13 +00:00
|
|
|
use Hydra::Helper::CatalystUtils;
|
2008-10-28 10:19:31 +00:00
|
|
|
|
2009-02-25 10:52:41 +00:00
|
|
|
|
|
|
|
# Put this controller at top-level.
|
2008-10-28 10:19:31 +00:00
|
|
|
__PACKAGE__->config->{namespace} = '';
|
|
|
|
|
|
|
|
|
2008-11-13 09:25:38 +00:00
|
|
|
sub begin :Private {
|
2008-11-26 17:43:45 +00:00
|
|
|
my ($self, $c) = @_;
|
2008-11-13 09:48:10 +00:00
|
|
|
$c->stash->{curUri} = $c->request->uri;
|
2009-03-23 13:52:24 +00:00
|
|
|
$c->stash->{version} = $ENV{"HYDRA_RELEASE"} || "<devel>";
|
2008-11-13 09:25:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-28 10:19:31 +00:00
|
|
|
sub index :Path :Args(0) {
|
2008-11-26 17:43:45 +00:00
|
|
|
my ($self, $c) = @_;
|
2009-04-02 16:15:57 +00:00
|
|
|
$c->stash->{template} = 'overview.tt';
|
|
|
|
$c->stash->{projects} = [$c->model('DB::Projects')->search({}, {order_by => 'displayname'})];
|
2008-11-26 23:49:51 +00:00
|
|
|
getBuildStats($c, $c->model('DB::Builds'));
|
2008-10-28 10:19:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-26 19:48:04 +00:00
|
|
|
sub login :Local {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
my $username = $c->request->params->{username} || "";
|
|
|
|
my $password = $c->request->params->{password} || "";
|
|
|
|
|
|
|
|
if ($username && $password) {
|
|
|
|
if ($c->authenticate({username => $username, password => $password})) {
|
|
|
|
$c->response->redirect(
|
|
|
|
defined $c->flash->{afterLogin}
|
|
|
|
? $c->flash->{afterLogin}
|
|
|
|
: $c->uri_for('/'));
|
|
|
|
return;
|
2008-11-26 23:25:24 +00:00
|
|
|
}
|
2008-11-26 19:48:04 +00:00
|
|
|
$c->stash->{errorMsg} = "Bad username or password.";
|
|
|
|
}
|
|
|
|
|
|
|
|
$c->stash->{template} = 'login.tt';
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub logout :Local {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
$c->logout;
|
|
|
|
$c->response->redirect($c->uri_for('/'));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-26 17:43:45 +00:00
|
|
|
sub queue :Local {
|
2008-11-26 19:48:04 +00:00
|
|
|
my ($self, $c) = @_;
|
2008-11-26 17:43:45 +00:00
|
|
|
$c->stash->{template} = 'queue.tt';
|
|
|
|
$c->stash->{queue} = [$c->model('DB::Builds')->search(
|
|
|
|
{finished => 0}, {join => 'schedulingInfo', order_by => ["priority DESC", "timestamp"]})];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-27 18:27:19 +00:00
|
|
|
sub getReleaseSet {
|
2008-11-27 21:08:17 +00:00
|
|
|
my ($c, $projectName, $releaseSetName) = @_;
|
2008-11-27 18:27:19 +00:00
|
|
|
|
2008-11-27 15:16:06 +00:00
|
|
|
my $project = $c->model('DB::Projects')->find($projectName);
|
2009-03-23 13:52:24 +00:00
|
|
|
notFound($c, "Project $projectName doesn't exist.") if !defined $project;
|
2009-03-13 15:41:19 +00:00
|
|
|
$c->stash->{project} = $project;
|
2008-11-27 15:16:06 +00:00
|
|
|
|
2008-12-16 16:26:33 +00:00
|
|
|
(my $releaseSet) = $c->model('DB::ReleaseSets')->find($projectName, $releaseSetName);
|
2009-03-23 13:52:24 +00:00
|
|
|
notFound($c, "Release set $releaseSetName doesn't exist.") if !defined $releaseSet;
|
2008-11-27 15:16:06 +00:00
|
|
|
$c->stash->{releaseSet} = $releaseSet;
|
|
|
|
|
|
|
|
(my $primaryJob) = $releaseSet->releasesetjobs->search({isprimary => 1});
|
2008-11-27 21:40:23 +00:00
|
|
|
#die "Release set $releaseSetName doesn't have a primary job." if !defined $primaryJob;
|
2008-11-27 18:27:19 +00:00
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
my $jobs = [$releaseSet->releasesetjobs->search({},
|
2008-11-27 18:27:19 +00:00
|
|
|
{order_by => ["isprimary DESC", "job", "attrs"]})];
|
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
$c->stash->{jobs} = $jobs;
|
2008-11-27 18:27:19 +00:00
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
return ($project, $releaseSet, $primaryJob, $jobs);
|
2008-11-27 18:27:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-27 21:08:17 +00:00
|
|
|
sub updateReleaseSet {
|
|
|
|
my ($c, $releaseSet) = @_;
|
|
|
|
|
|
|
|
my $releaseSetName = trim $c->request->params->{name};
|
2009-03-23 13:52:24 +00:00
|
|
|
error($c, "Invalid release set name: $releaseSetName")
|
|
|
|
unless $releaseSetName =~ /^[[:alpha:]][\w\-]*$/;
|
2008-11-27 21:08:17 +00:00
|
|
|
|
2009-03-09 16:22:41 +00:00
|
|
|
$releaseSet->update(
|
|
|
|
{ name => $releaseSetName
|
|
|
|
, description => trim $c->request->params->{description} });
|
2008-11-27 21:40:23 +00:00
|
|
|
|
|
|
|
$releaseSet->releasesetjobs->delete_all;
|
|
|
|
|
|
|
|
foreach my $param (keys %{$c->request->params}) {
|
|
|
|
next unless $param =~ /^job-(\d+)-name$/;
|
|
|
|
my $baseName = $1;
|
|
|
|
|
|
|
|
my $name = trim $c->request->params->{"job-$baseName-name"};
|
|
|
|
my $description = trim $c->request->params->{"job-$baseName-description"};
|
|
|
|
my $attrs = trim $c->request->params->{"job-$baseName-attrs"};
|
|
|
|
|
2009-03-20 15:19:17 +00:00
|
|
|
$name =~ /^([\w\-]+):([\w\-]+)$/ or error($c, "Invalid job name: $name");
|
2009-03-14 22:34:22 +00:00
|
|
|
my $jobsetName = $1;
|
|
|
|
my $jobName = $2;
|
|
|
|
|
|
|
|
error($c, "Jobset `$jobsetName' doesn't exist.")
|
|
|
|
unless $releaseSet->project->jobsets->find({name => $jobsetName});
|
|
|
|
|
|
|
|
# !!! We could check whether the job exists, but that would
|
|
|
|
# require the scheduler to have seen the job, which may not be
|
|
|
|
# the case.
|
2008-11-27 22:26:53 +00:00
|
|
|
|
2008-11-27 21:40:23 +00:00
|
|
|
$releaseSet->releasesetjobs->create(
|
2009-03-14 22:34:22 +00:00
|
|
|
{ jobset => $jobsetName
|
|
|
|
, job => $jobName
|
2008-11-27 21:40:23 +00:00
|
|
|
, description => $description
|
|
|
|
, attrs => $attrs
|
2008-12-01 11:15:03 +00:00
|
|
|
, isprimary => $c->request->params->{"primary"} eq $baseName ? 1 : 0
|
2008-11-27 21:40:23 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2009-03-23 13:52:24 +00:00
|
|
|
error($c, "There must be one primary job.")
|
|
|
|
if $releaseSet->releasesetjobs->search({isprimary => 1})->count != 1;
|
2008-11-27 21:08:17 +00:00
|
|
|
}
|
|
|
|
|
2008-12-15 13:12:51 +00:00
|
|
|
|
2008-11-27 18:27:19 +00:00
|
|
|
sub releases :Local {
|
2008-11-27 21:08:17 +00:00
|
|
|
my ($self, $c, $projectName, $releaseSetName, $subcommand) = @_;
|
2008-11-27 15:16:06 +00:00
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
my ($project, $releaseSet, $primaryJob, $jobs) = getReleaseSet($c, $projectName, $releaseSetName);
|
2008-11-27 21:08:17 +00:00
|
|
|
|
2008-11-28 18:54:09 +00:00
|
|
|
if (defined $subcommand && $subcommand ne "") {
|
2008-11-27 21:08:17 +00:00
|
|
|
|
2009-03-02 16:03:41 +00:00
|
|
|
requireProjectOwner($c, $project);
|
2008-11-27 17:01:41 +00:00
|
|
|
|
2008-11-27 21:08:17 +00:00
|
|
|
if ($subcommand eq "edit") {
|
|
|
|
$c->stash->{template} = 'edit-releaseset.tt';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
elsif ($subcommand eq "submit") {
|
|
|
|
$c->model('DB')->schema->txn_do(sub {
|
|
|
|
updateReleaseSet($c, $releaseSet);
|
|
|
|
});
|
|
|
|
return $c->res->redirect($c->uri_for("/releases", $projectName, $releaseSet->name));
|
|
|
|
}
|
|
|
|
|
2008-11-28 00:16:01 +00:00
|
|
|
elsif ($subcommand eq "delete") {
|
|
|
|
$c->model('DB')->schema->txn_do(sub {
|
|
|
|
$releaseSet->delete;
|
|
|
|
});
|
|
|
|
return $c->res->redirect($c->uri_for("/releasesets", $projectName));
|
|
|
|
}
|
|
|
|
|
2009-02-25 14:34:29 +00:00
|
|
|
else { error($c, "Unknown subcommand."); }
|
2008-11-27 21:08:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$c->stash->{template} = 'releases.tt';
|
2008-11-27 15:16:06 +00:00
|
|
|
|
|
|
|
my @releases = ();
|
2009-02-06 15:02:49 +00:00
|
|
|
push @releases, getRelease($_, $jobs) foreach getPrimaryBuildsForReleaseSet($project, $primaryJob);
|
2008-11-27 18:27:19 +00:00
|
|
|
$c->stash->{releases} = [@releases];
|
|
|
|
}
|
2008-11-27 15:16:06 +00:00
|
|
|
|
|
|
|
|
2008-11-28 00:16:01 +00:00
|
|
|
sub create_releaseset :Local {
|
|
|
|
my ($self, $c, $projectName, $subcommand) = @_;
|
|
|
|
|
|
|
|
my $project = $c->model('DB::Projects')->find($projectName);
|
2009-03-23 13:52:24 +00:00
|
|
|
error($c, "Project $projectName doesn't exist.") if !defined $project;
|
2009-03-13 15:41:19 +00:00
|
|
|
$c->stash->{project} = $project;
|
2008-11-28 00:16:01 +00:00
|
|
|
|
2009-03-02 16:03:41 +00:00
|
|
|
requireProjectOwner($c, $project);
|
2008-11-28 00:16:01 +00:00
|
|
|
|
|
|
|
if (defined $subcommand && $subcommand eq "submit") {
|
2009-02-25 14:34:29 +00:00
|
|
|
my $releaseSetName = $c->request->params->{name};
|
|
|
|
$c->model('DB')->schema->txn_do(sub {
|
|
|
|
# Note: $releaseSetName is validated in updateProject,
|
|
|
|
# which will abort the transaction if the name isn't
|
|
|
|
# valid.
|
|
|
|
my $releaseSet = $project->releasesets->create({name => $releaseSetName});
|
|
|
|
updateReleaseSet($c, $releaseSet);
|
|
|
|
return $c->res->redirect($c->uri_for("/releases", $projectName, $releaseSet->name));
|
|
|
|
});
|
2008-11-28 00:16:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$c->stash->{template} = 'edit-releaseset.tt';
|
|
|
|
$c->stash->{create} = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-27 18:27:19 +00:00
|
|
|
sub release :Local {
|
2008-11-27 21:08:17 +00:00
|
|
|
my ($self, $c, $projectName, $releaseSetName, $releaseId) = @_;
|
2008-11-27 18:27:19 +00:00
|
|
|
$c->stash->{template} = 'release.tt';
|
2008-11-27 15:16:06 +00:00
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
my ($project, $releaseSet, $primaryJob, $jobs) = getReleaseSet($c, $projectName, $releaseSetName);
|
2008-11-27 18:27:19 +00:00
|
|
|
|
2008-12-15 13:12:51 +00:00
|
|
|
if ($releaseId eq "latest") {
|
|
|
|
# Redirect to the latest successful release.
|
2009-02-06 15:02:49 +00:00
|
|
|
my $latest = getLatestSuccessfulRelease($project, $primaryJob, $jobs);
|
2009-02-25 14:34:29 +00:00
|
|
|
error($c, "This release set has no successful releases yet.") if !defined $latest;
|
2008-12-15 13:12:51 +00:00
|
|
|
return $c->res->redirect($c->uri_for("/release", $projectName, $releaseSetName, $latest->id));
|
|
|
|
}
|
|
|
|
|
2008-11-27 18:27:19 +00:00
|
|
|
# Note: we don't actually check whether $releaseId is a primary
|
|
|
|
# build, but who cares?
|
|
|
|
my $primaryBuild = $project->builds->find($releaseId,
|
2009-03-23 13:52:24 +00:00
|
|
|
{ join => 'resultInfo',
|
|
|
|
, '+select' => ["resultInfo.releasename", "resultInfo.buildstatus"]
|
|
|
|
, '+as' => ["releasename", "buildstatus"] })
|
|
|
|
or error($c, "Release $releaseId doesn't exist.");
|
2009-03-18 18:50:42 +00:00
|
|
|
|
2009-02-06 15:02:49 +00:00
|
|
|
$c->stash->{release} = getRelease($primaryBuild, $jobs);
|
2008-11-27 15:16:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-04 10:59:14 +00:00
|
|
|
# Hydra::Base::Controller::ListBuilds needs this.
|
|
|
|
sub get_builds : Chained('/') PathPart('') CaptureArgs(0) {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
$c->stash->{allBuilds} = $c->model('DB::Builds');
|
2009-03-13 17:32:08 +00:00
|
|
|
$c->stash->{allJobs} = $c->model('DB::Jobs');
|
2009-03-04 16:36:23 +00:00
|
|
|
$c->stash->{channelBaseName} = "everything";
|
2009-03-04 10:59:14 +00:00
|
|
|
}
|
|
|
|
|
2009-03-31 13:48:03 +00:00
|
|
|
|
|
|
|
sub robots_txt : Path('robots.txt') {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
2009-03-31 14:14:45 +00:00
|
|
|
sub uri_for {
|
|
|
|
my ($controller, $action, @args) = @_;
|
|
|
|
return $c->uri_for($c->controller($controller)->action_for($action), @args)->path;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub channelUris {
|
|
|
|
my ($controller, $bindings) = @_;
|
|
|
|
return
|
2009-03-31 14:55:47 +00:00
|
|
|
( uri_for($controller, 'closure', $bindings, "*")
|
|
|
|
, uri_for($controller, 'manifest', $bindings)
|
|
|
|
, uri_for($controller, 'nar', $bindings, "*")
|
|
|
|
, uri_for($controller, 'pkg', $bindings, "*")
|
|
|
|
, uri_for($controller, 'nixexprs', $bindings)
|
2009-03-31 14:14:45 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2009-03-31 13:48:03 +00:00
|
|
|
# Put actions that are expensive or not useful for indexing in
|
|
|
|
# robots.txt. Note: wildcards are not universally supported in
|
|
|
|
# robots.txt, but apparently Google supports them.
|
|
|
|
my @rules =
|
2009-03-31 14:55:47 +00:00
|
|
|
( uri_for('Build', 'buildtimedeps', ["*"])
|
|
|
|
, uri_for('Build', 'runtimedeps', ["*"])
|
|
|
|
, uri_for('Build', 'view_nixlog', ["*"], "*/tail")
|
2009-03-31 14:14:45 +00:00
|
|
|
, channelUris('Root', ["*"])
|
|
|
|
, channelUris('Project', ["*", "*"])
|
|
|
|
, channelUris('Jobset', ["*", "*", "*"])
|
|
|
|
, channelUris('Job', ["*", "*", "*", "*"])
|
|
|
|
, channelUris('Build', ["*"])
|
2009-03-31 13:48:03 +00:00
|
|
|
);
|
|
|
|
|
2009-03-31 14:55:47 +00:00
|
|
|
$c->stash->{'plain'} = { data => "User-agent: *\n" . join('', map { "Disallow: $_\n" } @rules) };
|
2009-03-31 13:48:03 +00:00
|
|
|
$c->forward('Hydra::View::Plain');
|
|
|
|
}
|
|
|
|
|
2009-03-04 10:59:14 +00:00
|
|
|
|
2008-10-28 10:19:31 +00:00
|
|
|
sub default :Path {
|
2008-11-26 17:43:45 +00:00
|
|
|
my ($self, $c) = @_;
|
2009-02-25 14:34:29 +00:00
|
|
|
notFound($c, "Page not found.");
|
2009-02-13 17:35:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-02-25 10:52:41 +00:00
|
|
|
sub end : ActionClass('RenderView') {
|
|
|
|
my ($self, $c) = @_;
|
|
|
|
|
|
|
|
if (scalar @{$c->error}) {
|
|
|
|
$c->stash->{template} = 'error.tt';
|
|
|
|
$c->stash->{errors} = $c->error;
|
2009-02-25 16:29:54 +00:00
|
|
|
if ($c->response->status >= 300) {
|
|
|
|
$c->stash->{httpStatus} =
|
|
|
|
$c->response->status . " " . HTTP::Status::status_message($c->response->status);
|
|
|
|
}
|
2009-02-25 10:52:41 +00:00
|
|
|
$c->clear_errors;
|
|
|
|
}
|
|
|
|
}
|
2008-10-28 10:19:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
1;
|