2009-03-13 15:23:08 +00:00
package Hydra::Controller::Jobset ;
use strict ;
use warnings ;
use base 'Hydra::Base::Controller::ListBuilds' ;
use Hydra::Helper::Nix ;
use Hydra::Helper::CatalystUtils ;
sub jobset : Chained('/') PathPart('jobset') CaptureArgs(2) {
my ( $ self , $ c , $ projectName , $ jobsetName ) = @ _ ;
my $ project = $ c - > model ( 'DB::Projects' ) - > find ( $ projectName )
or notFound ( $ c , "Project $projectName doesn't exist." ) ;
2009-03-13 15:41:19 +00:00
$ c - > stash - > { project } = $ project ;
2009-03-13 15:23:08 +00:00
2009-04-08 22:08:00 +00:00
$ c - > stash - > { jobset_ } = $ project - > jobsets - > search ( { name = > $ jobsetName } ) ;
$ c - > stash - > { jobset } = $ c - > stash - > { jobset_ } - > single
2009-03-13 15:23:08 +00:00
or notFound ( $ c , "Jobset $jobsetName doesn't exist." ) ;
}
2010-02-09 13:43:53 +00:00
sub jobsetIndex {
my ( $ self , $ c , $ forceStatus ) = @ _ ;
2009-03-13 15:23:08 +00:00
2009-04-02 16:15:57 +00:00
$ c - > stash - > { template } = 'jobset.tt' ;
2010-02-09 12:35:20 +00:00
#getBuildStats($c, scalar $c->stash->{jobset}->builds);
2009-04-02 16:15:57 +00:00
2010-02-09 13:47:20 +00:00
my $ projectName = $ c - > stash - > { project } - > name ;
my $ jobsetName = $ c - > stash - > { jobset } - > name ;
# Get the active / inactive jobs in this jobset.
my @ jobs = $ c - > stash - > { jobset } - > jobs - > search (
{ } ,
{ select = > [
"name" ,
\ ( "exists (select 1 from builds where project = '$projectName' and jobset = '$jobsetName' and job = me.name and isCurrent = 1) as active" )
]
, as = > [ "name" , "active" ]
, order_by = > [ "name" ] } ) ;
$ c - > stash - > { activeJobs } = [] ;
$ c - > stash - > { inactiveJobs } = [] ;
foreach my $ job ( @ jobs ) {
if ( $ job - > get_column ( 'active' ) ) {
push @ { $ c - > stash - > { activeJobs } } , $ job - > name ;
} else {
push @ { $ c - > stash - > { inactiveJobs } } , $ job - > name ;
}
}
2010-02-26 09:14:49 +00:00
$ c - > stash - > { systems } = [ $ c - > stash - > { jobset } - > builds - > search ( { iscurrent = > 1 } , { select = > [ "system" ] , distinct = > 1 , order_by = > "system" } ) ] ;
2010-02-05 14:48:22 +00:00
# status per system
my @ systems = ( ) ;
foreach my $ system ( @ { $ c - > stash - > { systems } } ) {
push ( @ systems , $ system - > system ) ;
}
2010-02-09 13:43:53 +00:00
if ( $ forceStatus || scalar ( @ { $ c - > stash - > { activeJobs } } ) <= 20 ) {
2010-02-09 13:47:20 +00:00
my @ select = ( ) ;
my @ as = ( ) ;
push ( @ select , "job" ) ; push ( @ as , "job" ) ;
foreach my $ system ( @ systems ) {
2010-03-01 10:10:20 +00:00
push ( @ select , "(SELECT buildstatus FROM BuildResultInfo bri NATURAL JOIN Builds b WHERE b.id = (SELECT MAX(id) FROM Builds t WHERE t.project = me.project AND t.jobset = me.jobset AND t.job = me.job AND t.system = '$system' AND t.iscurrent = 1 ))" ) ;
2010-02-09 13:47:20 +00:00
push ( @ as , $ system ) ;
2010-03-01 10:10:20 +00:00
push ( @ select , "(SELECT b.id FROM BuildResultInfo bri NATURAL JOIN Builds b WHERE b.id = (SELECT MAX(id) FROM Builds t WHERE t.project = me.project AND t.jobset = me.jobset AND t.job = me.job AND t.system = '$system' AND t.iscurrent = 1 ))" ) ;
2010-02-09 13:47:20 +00:00
push ( @ as , $ system . "-build" ) ;
}
$ c - > stash - > { activeJobsStatus } =
[ $ c - > model ( 'DB' ) - > resultset ( 'ActiveJobsForJobset' ) - > search (
{ } ,
{ bind = > [ $ c - > stash - > { project } - > name , $ c - > stash - > { jobset } - > name ]
, select = > \ @ select
, as = > \ @ as
, order_by = > [ "job" ]
} ) ] ;
}
2010-02-05 14:48:22 +00:00
# last builds for jobset
my $ tmp = $ c - > stash - > { jobset } - > builds ;
$ c - > stash - > { lastBuilds } = [ joinWithResultInfo ( $ c , $ tmp )
- > search ( { finished = > 1 } , { order_by = > "timestamp DESC" , rows = > 5 } ) ] ;
2009-03-13 15:23:08 +00:00
}
2010-02-09 13:43:53 +00:00
sub index : Chained('jobset') PathPart('') Args(0) {
my ( $ self , $ c ) = @ _ ;
jobsetIndex ( $ self , $ c , 0 ) ;
}
sub indexWithStatus : Chained('jobset') PathPart('') Args(1) {
my ( $ self , $ c , $ forceStatus ) = @ _ ;
jobsetIndex ( $ self , $ c , 1 ) ;
}
2009-03-13 15:23:08 +00:00
# Hydra::Base::Controller::ListBuilds needs this.
sub get_builds : Chained('jobset') PathPart('') CaptureArgs(0) {
my ( $ self , $ c ) = @ _ ;
2009-03-13 17:32:08 +00:00
$ c - > stash - > { allBuilds } = $ c - > stash - > { jobset } - > builds ;
2009-04-03 15:37:21 +00:00
$ c - > stash - > { jobStatus } = $ c - > model ( 'DB' ) - > resultset ( 'JobStatusForJobset' )
- > search ( { } , { bind = > [ $ c - > stash - > { project } - > name , $ c - > stash - > { jobset } - > name ] } ) ;
2009-04-08 22:08:00 +00:00
$ c - > stash - > { allJobsets } = $ c - > stash - > { jobset_ } ;
$ c - > stash - > { allJobs } = $ c - > stash - > { jobset } - > jobs ;
2009-04-03 15:37:21 +00:00
$ c - > stash - > { latestSucceeded } = $ c - > model ( 'DB' ) - > resultset ( 'LatestSucceededForJobset' )
- > search ( { } , { bind = > [ $ c - > stash - > { project } - > name , $ c - > stash - > { jobset } - > name ] } ) ;
2009-03-13 15:23:08 +00:00
$ c - > stash - > { channelBaseName } =
2009-03-13 15:41:19 +00:00
$ c - > stash - > { project } - > name . "-" . $ c - > stash - > { jobset } - > name ;
2009-03-13 15:23:08 +00:00
}
2009-04-02 16:15:57 +00:00
sub edit : Chained('jobset') PathPart Args(0) {
my ( $ self , $ c ) = @ _ ;
requireProjectOwner ( $ c , $ c - > stash - > { project } ) ;
$ c - > stash - > { template } = 'jobset.tt' ;
$ c - > stash - > { edit } = 1 ;
}
sub submit : Chained('jobset') PathPart Args(0) {
my ( $ self , $ c ) = @ _ ;
requireProjectOwner ( $ c , $ c - > stash - > { project } ) ;
requirePost ( $ c ) ;
2009-04-22 22:43:04 +00:00
txn_do ( $ c - > model ( 'DB' ) - > schema , sub {
2009-04-02 16:15:57 +00:00
updateJobset ( $ c , $ c - > stash - > { jobset } ) ;
} ) ;
$ c - > res - > redirect ( $ c - > uri_for ( $ self - > action_for ( "index" ) ,
[ $ c - > stash - > { project } - > name , $ c - > stash - > { jobset } - > name ] ) ) ;
}
2010-06-04 14:43:28 +00:00
sub hide : Chained('jobset') PathPart Args(0) {
my ( $ self , $ c ) = @ _ ;
requireProjectOwner ( $ c , $ c - > stash - > { project } ) ;
txn_do ( $ c - > model ( 'DB' ) - > schema , sub {
$ c - > stash - > { jobset } - > update ( { hidden = > 1 , enabled = > 0 } ) ;
} ) ;
$ c - > res - > redirect ( $ c - > uri_for ( $ c - > controller ( 'Project' ) - > action_for ( "view" ) ,
[ $ c - > stash - > { project } - > name ] ) ) ;
}
sub unhide : Chained('jobset') PathPart Args(0) {
my ( $ self , $ c ) = @ _ ;
requireProjectOwner ( $ c , $ c - > stash - > { project } ) ;
txn_do ( $ c - > model ( 'DB' ) - > schema , sub {
$ c - > stash - > { jobset } - > update ( { hidden = > 0 } ) ;
} ) ;
$ c - > res - > redirect ( $ c - > uri_for ( $ c - > controller ( 'Project' ) - > action_for ( "view" ) ,
[ $ c - > stash - > { project } - > name ] ) ) ;
}
2009-04-02 16:15:57 +00:00
sub delete : Chained('jobset') PathPart Args(0) {
my ( $ self , $ c ) = @ _ ;
requireProjectOwner ( $ c , $ c - > stash - > { project } ) ;
requirePost ( $ c ) ;
2009-04-22 22:43:04 +00:00
txn_do ( $ c - > model ( 'DB' ) - > schema , sub {
2009-04-02 16:15:57 +00:00
$ c - > stash - > { jobset } - > delete ;
} ) ;
$ c - > res - > redirect ( $ c - > uri_for ( $ c - > controller ( 'Project' ) - > action_for ( "view" ) ,
[ $ c - > stash - > { project } - > name ] ) ) ;
}
2009-10-26 15:39:14 +00:00
sub nixExprPathFromParams {
my ( $ c ) = @ _ ;
2009-04-02 16:15:57 +00:00
# The Nix expression path must be relative and can't contain ".." elements.
my $ nixExprPath = trim $ c - > request - > params - > { "nixexprpath" } ;
error ( $ c , "Invalid Nix expression path: $nixExprPath" ) if $ nixExprPath !~ /^$relPathRE$/ ;
my $ nixExprInput = trim $ c - > request - > params - > { "nixexprinput" } ;
error ( $ c , "Invalid Nix expression input name: $nixExprInput" ) unless $ nixExprInput =~ /^\w+$/ ;
2009-10-26 15:39:14 +00:00
return ( $ nixExprPath , $ nixExprInput ) ;
}
sub checkInput {
my ( $ c , $ baseName ) = @ _ ;
my $ inputName = trim $ c - > request - > params - > { "input-$baseName-name" } ;
error ( $ c , "Invalid input name: $inputName" ) unless $ inputName =~ /^[[:alpha:]]\w*$/ ;
my $ inputType = trim $ c - > request - > params - > { "input-$baseName-type" } ;
error ( $ c , "Invalid input type: $inputType" ) unless
2010-07-27 16:09:35 +00:00
$ inputType eq "svn" || $ inputType eq "svn-checkout" || $ inputType eq "hg" || $ inputType eq "tarball" ||
2009-10-26 15:39:14 +00:00
$ inputType eq "string" || $ inputType eq "path" || $ inputType eq "boolean" ||
2010-01-19 14:15:31 +00:00
$ inputType eq "git" || $ inputType eq "build" || $ inputType eq "sysbuild" ;
2009-10-26 15:39:14 +00:00
return ( $ inputName , $ inputType ) ;
}
sub checkInputValue {
my ( $ c , $ type , $ value ) = @ _ ;
$ value = trim $ value ;
error ( $ c , "Invalid Boolean value: $value" ) if
$ type eq "boolean" && ! ( $ value eq "true" || $ value eq "false" ) ;
return $ value ;
}
sub updateJobset {
my ( $ c , $ jobset ) = @ _ ;
my $ jobsetName = trim $ c - > request - > params - > { "name" } ;
error ( $ c , "Invalid jobset name: $jobsetName" ) unless $ jobsetName =~ /^[[:alpha:]][\w\-]*$/ ;
my ( $ nixExprPath , $ nixExprInput ) = nixExprPathFromParams $ c ;
2009-04-02 16:15:57 +00:00
$ jobset - > update (
{ name = > $ jobsetName
, description = > trim ( $ c - > request - > params - > { "description" } )
, nixexprpath = > $ nixExprPath
, nixexprinput = > $ nixExprInput
2009-10-08 11:39:16 +00:00
, enabled = > trim ( $ c - > request - > params - > { enabled } ) eq "1" ? 1 : 0
2010-01-06 13:07:59 +00:00
, enableemail = > trim ( $ c - > request - > params - > { enableemail } ) eq "1" ? 1 : 0
, emailoverride = > trim ( $ c - > request - > params - > { emailoverride } )
2010-08-10 06:48:45 +00:00
, keepnr = > trim ( $ c - > request - > params - > { keepnr } )
2009-04-02 16:15:57 +00:00
} ) ;
my % inputNames ;
# Process the inputs of this jobset.
foreach my $ param ( keys % { $ c - > request - > params } ) {
next unless $ param =~ /^input-(\w+)-name$/ ;
2009-10-26 15:39:14 +00:00
my $ baseName = $ 1 ;
next if $ baseName eq "template" ;
2009-04-02 16:15:57 +00:00
2009-10-26 15:39:14 +00:00
my ( $ inputName , $ inputType ) = checkInput ( $ c , $ baseName ) ;
2009-04-02 16:15:57 +00:00
$ inputNames { $ inputName } = 1 ;
my $ input ;
2009-10-26 15:39:14 +00:00
if ( $ baseName =~ /^\d+$/ ) { # numeric base name is auto-generated, i.e. a new entry
2009-04-02 16:15:57 +00:00
$ input = $ jobset - > jobsetinputs - > create (
{ name = > $ inputName
, type = > $ inputType
} ) ;
} else { # it's an existing input
2009-10-26 15:39:14 +00:00
$ input = ( $ jobset - > jobsetinputs - > search ( { name = > $ baseName } ) ) [ 0 ] ;
2009-04-02 16:15:57 +00:00
die unless defined $ input ;
$ input - > update ( { name = > $ inputName , type = > $ inputType } ) ;
}
# Update the values for this input. Just delete all the
# current ones, then create the new values.
$ input - > jobsetinputalts - > delete_all ;
2009-10-26 15:39:14 +00:00
my $ values = $ c - > request - > params - > { "input-$baseName-values" } ;
2009-04-02 16:15:57 +00:00
$ values = [] unless defined $ values ;
$ values = [ $ values ] unless ref ( $ values ) eq 'ARRAY' ;
my $ altnr = 0 ;
foreach my $ value ( @ { $ values } ) {
2009-10-26 15:39:14 +00:00
$ value = checkInputValue ( $ c , $ inputType , $ value ) ;
2009-04-02 16:15:57 +00:00
$ input - > jobsetinputalts - > create ( { altnr = > $ altnr + + , value = > $ value } ) ;
}
}
# Get rid of deleted inputs.
my @ inputs = $ jobset - > jobsetinputs - > all ;
foreach my $ input ( @ inputs ) {
$ input - > delete unless defined $ inputNames { $ input - > name } ;
}
}
2010-04-23 11:20:40 +00:00
sub clone : Chained('jobset') PathPart('clone') Args(0) {
my ( $ self , $ c ) = @ _ ;
my $ jobset = $ c - > stash - > { jobset } ;
requireProjectOwner ( $ c , $ jobset - > project ) ;
$ c - > stash - > { template } = 'clone-jobset.tt' ;
}
sub clone_submit : Chained('jobset') PathPart('clone/submit') Args(0) {
my ( $ self , $ c ) = @ _ ;
my $ jobset = $ c - > stash - > { jobset } ;
requireProjectOwner ( $ c , $ jobset - > project ) ;
requirePost ( $ c ) ;
my $ newjobsetName = trim $ c - > request - > params - > { "newjobset" } ;
error ( $ c , "Invalid jobset name: $newjobsetName" ) unless $ newjobsetName =~ /^[[:alpha:]][\w\-]*$/ ;
my $ newjobset ;
txn_do ( $ c - > model ( 'DB' ) - > schema , sub {
$ newjobset = $ jobset - > project - > jobsets - > create (
{ name = > $ newjobsetName
, description = > $ jobset - > description
, nixexprpath = > $ jobset - > nixexprpath
, nixexprinput = > $ jobset - > nixexprinput
, enabled = > 0
, enableemail = > $ jobset - > enableemail
, emailoverride = > $ jobset - > emailoverride
} ) ;
foreach my $ input ( $ jobset - > jobsetinputs ) {
my $ newinput = $ newjobset - > jobsetinputs - > create ( { name = > $ input - > name , type = > $ input - > type } ) ;
foreach my $ inputalt ( $ input - > jobsetinputalts ) {
$ newinput - > jobsetinputalts - > create ( { altnr = > $ inputalt - > altnr , value = > $ inputalt - > value } ) ;
}
}
} ) ;
$ c - > res - > redirect ( $ c - > uri_for ( $ c - > controller ( 'Jobset' ) - > action_for ( "edit" ) , [ $ jobset - > project - > name , $ newjobsetName ] ) ) ;
}
2009-03-13 15:23:08 +00:00
1 ;