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:25:38 +00:00
|
|
|
$c->stash->{projects} = [$c->model('DB::Projects')->search({}, {order_by => 'displayname'})];
|
2008-11-13 09:48:10 +00:00
|
|
|
$c->stash->{curUri} = $c->request->uri;
|
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) = @_;
|
2008-10-28 10:19:31 +00:00
|
|
|
$c->stash->{template} = 'index.tt';
|
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 15:16:06 +00:00
|
|
|
sub releasesets :Local {
|
|
|
|
my ($self, $c, $projectName) = @_;
|
|
|
|
$c->stash->{template} = 'releasesets.tt';
|
|
|
|
|
|
|
|
my $project = $c->model('DB::Projects')->find($projectName);
|
2009-02-25 14:34:29 +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
|
|
|
|
|
|
|
$c->stash->{releaseSets} = [$project->releasesets->all];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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);
|
2008-11-27 18:27:19 +00:00
|
|
|
die "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);
|
2008-11-27 21:08:17 +00:00
|
|
|
die "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};
|
|
|
|
die "Invalid release set name: $releaseSetName" unless $releaseSetName =~ /^[[:alpha:]]\w*$/;
|
|
|
|
|
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-14 22:34:22 +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(
|
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
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
die "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);
|
|
|
|
die "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,
|
|
|
|
{ join => 'resultInfo', '+select' => ["resultInfo.releasename"], '+as' => ["releasename"] });
|
2009-02-25 14:34:29 +00:00
|
|
|
error($c, "Release $releaseId doesn't exist.") if !defined $primaryBuild;
|
2008-11-27 18:27:19 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
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;
|