293 lines
9.2 KiB
Perl
Raw Normal View History

2008-11-25 11:01:42 +00:00
package Hydra::Controller::Root;
use strict;
use warnings;
use base 'Hydra::Base::Controller::ListBuilds';
2008-11-25 11:01:42 +00:00
use Hydra::Helper::Nix;
use Hydra::Helper::CatalystUtils;
# Put this controller at top-level.
__PACKAGE__->config->{namespace} = '';
2008-11-13 09:25:38 +00:00
sub begin :Private {
my ($self, $c) = @_;
2008-11-13 09:48:10 +00:00
$c->stash->{curUri} = $c->request->uri;
$c->stash->{version} = $ENV{"HYDRA_RELEASE"} || "<devel>";
2008-11-13 09:25:38 +00:00
}
sub index :Path :Args(0) {
my ($self, $c) = @_;
$c->stash->{template} = 'overview.tt';
$c->stash->{projects} = [$c->model('DB::Projects')->search({}, {order_by => 'displayname'})];
getBuildStats($c, $c->model('DB::Builds'));
}
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 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('/'));
}
sub queue :Local {
2008-11-26 19:48:04 +00:00
my ($self, $c) = @_;
$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
my $project = $c->model('DB::Projects')->find($projectName);
notFound($c, "Project $projectName doesn't exist.") if !defined $project;
2009-03-13 15:41:19 +00:00
$c->stash->{project} = $project;
(my $releaseSet) = $c->model('DB::ReleaseSets')->find($projectName, $releaseSetName);
notFound($c, "Release set $releaseSetName doesn't exist.") if !defined $releaseSet;
$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
my $jobs = [$releaseSet->releasesetjobs->search({},
2008-11-27 18:27:19 +00:00
{order_by => ["isprimary DESC", "job", "attrs"]})];
$c->stash->{jobs} = $jobs;
2008-11-27 18:27:19 +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};
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");
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(
{ 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
});
}
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-11-27 18:27:19 +00:00
sub releases :Local {
2008-11-27 21:08:17 +00:00
my ($self, $c, $projectName, $releaseSetName, $subcommand) = @_;
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
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';
my @releases = ();
push @releases, getRelease($_, $jobs) foreach getPrimaryBuildsForReleaseSet($project, $primaryJob);
2008-11-27 18:27:19 +00:00
$c->stash->{releases} = [@releases];
}
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);
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
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';
my ($project, $releaseSet, $primaryJob, $jobs) = getReleaseSet($c, $projectName, $releaseSetName);
2008-11-27 18:27:19 +00:00
if ($releaseId eq "latest") {
# Redirect to the latest successful release.
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;
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,
{ 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
$c->stash->{release} = getRelease($primaryBuild, $jobs);
}
# 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');
$c->stash->{channelBaseName} = "everything";
}
2009-03-31 13:48:03 +00:00
sub robots_txt : Path('robots.txt') {
my ($self, $c) = @_;
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 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")
, 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');
}
sub default :Path {
my ($self, $c) = @_;
2009-02-25 14:34:29 +00:00
notFound($c, "Page not found.");
}
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);
}
$c->clear_errors;
}
}
1;